首页 > 代码库 > 15.C#回顾及匿名类型(八章8.1-8.5)

15.C#回顾及匿名类型(八章8.1-8.5)

  今天的篇幅应该会很长,除了回顾前面学的一些,还有写一些关于匿名类型的相关知识,总体上对后续的学习很有帮助,学好了,后面更容易理解,不明白的,那就前面多翻几次,看多了总是会理解的。那么,进入正题吧。

  • 自动实现属性

  我们的很多工作都是由编译器帮我们去完成,如我们要说的自动实现属性。使用自动实现属性时,C#3执行了一个简单的编译转换,在类的内部生成一个私有的字段,使用不友好的命名(防止命名冲突)。在C#2中允许为取值和赋值方法指定不同的访问权限,现在我们还可以创建静态的自动属性。

  • 隐式类型

  使用隐式类型,在编写代码时,没有显式地声明类型,但在编译的结果中,编译器会获取初始化表达式的类型,使用变量也具有该类型。隐式类型只有在以下几种情况才能使用:

  1. 被声明的变量是一个局部变量,而不是一个静态字段和实例字段
  2. 变量声明的同时需要初始化
  3. 初始化表达式不能是方法组,也不是匿名函数
  4. 初始化表达式不是null
  5. 语句中只声明了一个变量
  6. 初始化表达式不能包含正在使用的变量
  •  初始化

  一种方法是使用无参的构造函数先实例化一个对象,然后分别为每个公开的属性赋值。另一种则是在构造函数中将属性值作为参数,在构造函数中为每个属性赋值,这里可以给公开和私有的赋值。使用自动实现属性,则可以使用对象初始化器,如下,三个类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 User12 {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 };

  匿名类型包含以下成员:

  1. 一个获取所有初始值的构造函数
  2. 公有的只读属性
  3. 属性的私有字段
  4. 重写了由object中继承的方法

  关于投影初始化程序,简单地理解从一个集合中,抽取集合元素中的各别属性,组成一个匿名类型,从而返回一个包含匿名类型的集合,这也是为什么我们使用var关键字,因为我们真的不知道返回的类型,使用var让编译器替我们理解返回类型,那问题来了,这个匿名类型的声明是否由编译器帮我们生成?它帮我们生成了类型,使用反编译工具就能知晓。

  请斧正。

 

15.C#回顾及匿名类型(八章8.1-8.5)