今天的篇幅应该会很长,除了回顾前面学的一些,还有写一些关于匿名类型的相关知识,总体上对后续的学习很有帮助,学好了,后面更容易理解,不明白的,那就前面多翻几次,看多了总是会理解的。那么,进入正题吧。
我们的很多工作都是由编译器帮我们去完成,如我们要说的自动实现属性。使用自动实现属性时,C#3执行了一个简单的编译转换,在类的内部生成一个私有的字段,使用不友好的命名(防止命名冲突)。在C#2中允许为取值和赋值方法指定不同的访问权限,现在我们还可以创建静态的自动属性。
使用隐式类型,在编写代码时,没有显式地声明类型,但在编译的结果中,编译器会获取初始化表达式的类型,使用变量也具有该类型。隐式类型只有在以下几种情况才能使用:
一种方法是使用无参的构造函数先实例化一个对象,然后分别为每个公开的属性赋值。另一种则是在构造函数中将属性值作为参数,在构造函数中为每个属性赋值,这里可以给公开和私有的赋值。使用自动实现属性,则可以使用对象初始化器,如下,三个类Computer、Mouse、User
1 class Computer 2 { 3 public string Cpu { get; set; } 4 public Mouse Mouse { get; set; } 5 public List<User> Users { get; set; } 6 } 7 class Mouse 8 { 9 public string Brand { get; set; } 10 } 11 class User 12 { 13 public string Name { get; set; } 14 }
上述三个类都没有构造函数(有一个默认的无参构造函数),使用对象初始化器就能很方便的去实例化对象。
1 Computer c = new Computer() 2 { 3 Cpu = "AMD", 4 Mouse = new Mouse() { Brand = "罗技" }, 5 Users = new List<User> { 6 new User() {Name="小A" }, 7 new User() {Name="小B" }, 8 new User() {Name="小C" } 9 } 10 };
可以看到上述的每一个对象实例都使用了对象初始器来实例化对象,调用的构造函数都是系统默认的构造函数,当我们将无参的构造函数设置为私有时,上述语句将无效。那我们就可以想像,当有一个为Cup赋值的构造函数,则在new Computer(cupName)接大括号"{}"来初始化对象,如
1 class Computer 2 { 3 public string Cpu { get; set; } 4 public Mouse Mouse { get; set; } 5 public List<User> Users { get; set; } 6 public Computer(string cpu) { 7 this.Cpu = cpu; 8 } 9 public Computer() 10 { 11 12 } 13 } 14 15 Computer c1 = new Computer("AMD") 16 { 17 Mouse = new Mouse() { Brand = "罗技" }, 18 Users = new List<User> { 19 new User() {Name="小A" }, 20 new User() {Name="小B" }, 21 new User() {Name="小C" } 22 } 23 };
注:调用无参的构造函数时,使用省略类名后面的括号。
是不是看上面的代码不断是很多,那我们可以再精简一下new List<User>,和new Mouse。
1 Computer c1 = new Computer("AMD") 2 { 3 Mouse = { Brand = "罗技" }, 4 Users = { 5 new User() {Name="小A" }, 6 new User() {Name="小B" }, 7 new User() {Name="小C" } 8 } 9 };
我们直接将类型名称给去除了,看到这里是否也想到可以把new User也给去除了,不过我试过是不行的,应该是无法确定要转换的类型吧。这个有点类似在C++11中统一使用"{}"来初始化对象。关于集合初始化可以参照上述中的对属性Users的初始化,而上述中Mouse = { Brand = "罗技" }则称呼为初始化嵌入对象。
在C#1和C#2中,数组的声明和初始化如下
1 string[] names = { "a", "b", "c" };
如果一个方法的签名如下:
1 public void Method0(string[] names)
那使用大括号中的表达式不能作为参数传入该方法,如
1 Method0({ "a", "b", "c" });
必须要告诉编译器传入的数组是什么类型的数组,如
1 Method0(new string[] { "a", "b", "c" });
但如果我们显式的指定类型,可以让编译器自己推断,则用到了隐式类型数组,里面也有涉及到协变性,如果A继承于B,参数为A的数组,那么我们使用隐式类型的数组,就可以传入A实例和B实例的数组。如:
1 class A 2 { 3 } 4 5 class B : A 6 { 7 } 8 9 public static void Method1(A[] args) 10 { 11 12 } 13 14 A a = new A(); 15 B b = new B(); 16 Method1(new[] { a, b }); 17 Method1(new[] { new A(), new B() });
匿名类型常用作用于LINQ中返回一系列的没有具体类型名的对象,也可以使用单独使用(在不想创建多余的类时)。
1 var p = new { Name = "a", Age = 12 };
接下来,就可以使用变量p,p有两个属性Name="a"和Age=12,也可以使用匿名类型来初始化数组,如
1 var ps = new[] { 2 new { Name = "a", Age = 12}, 3 new { Name = "a", Age = 12}, 4 new { Name = "a", Age = 12} 5 };
匿名类型包含以下成员:
关于投影初始化程序,简单地理解从一个集合中,抽取集合元素中的各别属性,组成一个匿名类型,从而返回一个包含匿名类型的集合,这也是为什么我们使用var关键字,因为我们真的不知道返回的类型,使用var让编译器替我们理解返回类型,那问题来了,这个匿名类型的声明是否由编译器帮我们生成?它帮我们生成了类型,使用反编译工具就能知晓。
请斧正。