上面我们向读者讲解了隐式类型,接下来我们继续来讨论C#3.0另一个新特性:匿名类型。
20.2.1 引入匿名类型
在很多情况下,我们需要一种能够临时将一批具有一定关联的数据存放起来的对象;或者在某些情况下,我们对仅一个对象的“形状”(如属性的名字和类型等)比较感兴趣。例如Book类,当它和其他商品放在一起进行查询时,我们可能仅对其名称和价格感兴趣,并且希望将这两种属性放在另外一个单独的临时对象中以备今后使用。这时,我们关注的仅仅是这个临时对象具有Name和Price的属性感兴趣,至于它究竟是什么类型就无关紧要了。然而,为了使这样一个对象得以存在,我们不得不为这个无关紧要的类型写上一大堆“样本代码”,无非就是定义一个如BookAsGood的类,其中无非也就是形如m_name和m_price的私有域和名为Name与Price的公共可读写方法。代码如下所示:
public class BookAsGood
{
// 定义一组私有成员变量
private string m_name;
private double m_price;
// 为成员变量设置属性
public string Name
{
get
{
return this.m_name;
}
set
{
this.m_name = value;
}
}
public string Price
{
get
{
return this.m_price;
}
set
{
this.m_price = value;
}
}
}
如果像这样封装的成员太多,代码量还是很可怕的,并且维护的工作量也相当大。针对于这些问题,在C# 3.0中,我们有了一个解决这种问题的捷径,称之为匿名类型,它是C#匿名方法语法的扩展。
20.2.2 创建和使用引入匿名类型
var与new关键字一起使用时,可以创建匿名类型。匿名类型只是一个继承了object的、没有名称的类。该类的定义从初始化器中推断,类似于隐式类型化的变量。看下面的例子:
如果需要一个对象包含名字和价格,则可以声明如下:
// 代表一些书的一个匿名对象
var book = new { Name = "C#", Price = 100};
这会生成一个包含Name和Price属性的对象。如果创建另一个对象,如下所示:
var bookA = new { Name = "C# 3.0", Price = 80 };
其中,book和bookA的类型就是相同的。例如,我们可以这样设置:
book = bookA;
如果所设置的值来自于另一个对象,初始化器就可以简化。如果已经有一个包含Name和Price属性的类,且有一个该类的实例b,则book对象就可以初始化为:
BookAsGood b = new BookAsGood();
var book = new { b.Name, b.Price };
在上面的代码中,b对象的属性名应投射为新对象名book。所以book对象应有Name和Price属性。
定义好之后,我们就可以这样来访问它的属性,如:
// 像普通实例那样访问其中的属性
Console.WriteLine(book.Name);
如果换成这样访问呢?如:
book.Name = "C# 3.0";
编译器会报错:无法对属性或索引器“AnonymousType#1.Name”赋值 -- 它是只读的。因为在这里book的各个属性只实现了get而没有实现set。所谓“匿名类型”,可以看到,我们有了一个叫做book的实例,但却没有它的强类型名称,在定义的时候将其定义为隐式类型(事实上这也是强制的)。在给它指定属性的时候也采用的对象初始化器的语法。在实际运行的时候,C#编译器会为它生成一个隐藏的类型名称,我们在程序中是不能对它进行访问的。
一个匿名类型也是它的生存周期,那就是在生成它的方法之内。如果要向另一个方法以参数的形式传递它,那就必须先将其转换为object类型(匿名类型也是从object直接派生而来的)。但是这样会毁坏它内部属性的强类型。如果需要保持内部属性的强类型,所以建议在大多数情况下,还是采用传统的类或者结构体来存储数据比较可靠。
20.2.3 匿名类型与隐式类型变量的区别
l 隐式类型变量是指我们可以通过等号右边的表达式,推断出等号左边该是那种类型。如:
var Name = "C#";
我们可以根据等号右边的表达式“C#”,推断出等号左边的变量Name是string类型。
l 匿名类型则是指根据这个类型的初始化函数,我们可以推导出和创建出这个类型的实例。这两个特性很多时候是一起作用的。如:
var book = new { Name = "C#", Price = 100};
上面的book就是一个匿名类型。其中的Name = "C#"和Price = 100又可以分别看作是隐式类型变量。