自动属性的增强
- 只读自动属性
以前自动属性必须同时提供setter和getter方法,因而只读属性只能通过先声明field,然后property只提供getter方法来实现,无法通过自动属性来实现。在c#6.0中,可以通过如下的形式声明只读的自动属性:
public string FirstName { get; }
对于只读的自动属性,其backing field是readonly的,其值可以在属性初始化器(下面会详细讲解属性初始化器)或者类的构造函数中给予。在下面的例子中FirstName和LastName属性是通过自动属性初始化器赋值的,而TimeOfStartingWrite是通过构造函数赋值的。
public class Author { public DateTime TimeOfStartingWrite { get; } public string FirstName { get; } = "Dery"; public string LastName { get; } = "Xu"; public Author(DateTime timeOfStartingWrite) { TimeOfStartingWrite = timeOfStartingWrite; } }
由于只读属性没有setter方法,它的值是通过直接赋给其backing field的。
- 自动属性初始化器
在上面的例子中可以看到对于自动属性,我们可以使用属性初始化器来为其赋值。例子如下:
public class Book { public int Number { get;}=100; public string Abstract { get; set; }="this is abstract"; public string Name { get; set; } public float Price { get; set; } public Author PrimaryAuthor { get; set; } public List<Author> Authors { get; set; } }
其中Number是一个只读自动属性,我们通过属性初始化器为其赋值100,而Abstract是一个可读写的自动属性,我们通过属性初始化器为其赋值为字符串"this is abstract"。自动属性初始化器直接赋值给后台生成的field,不会走自动属性的setter方法。它和field初始化器一样具有如下的一些特点:
- 按照书写顺序执行,所以在上例中,先执行Number属性的初始化,再执行Abstract属性的初始化
- 不能使用this,因为它是在对象完全初始化之前运行的
自动属性初始化器中可用来赋给属性的值似乎很有限,而下面的主构造函数可以帮助它获得跟多可能的值。
主构造函数
主构造函数功能把构造函数的声明合并到了类的声明中。让我们可以编写如下所示的代码:
[Serializable] public class Patent(string title, string yearOfPublication) { public Patent(string title, string yearOfPublication, IEnumerable<string> inventors) :this(title, yearOfPublication) { Inventors.AddRange(inventors); } private string _Title = title; public string Title { get { return _Title; } set { if (value == null) { throw new ArgumentNullException("Title"); } _Title = value; } } }
然而,这次的release中还不支持该功能,会在以后的更新中进一步说明。
表达式体函数和属性
对于Lambda表达式,我们知道可以通过表达式体或者在语句块中的普通函数来声明,现在这个功能也可以应用到类的函数成员上。
- 函数成员
public int Add(int op1, int op2) => op1 + op2; public void Print(string message) => Console.WriteLine("Hello " + message);
使用表达式体函数的效果和只有单一return语句的函数体一样,所以上面的例子和下面的code是一样的
public int Add(int op1, int op2) { return op1 + op2; } public void Print(string message) { Console.WriteLine("Hello " + message); }
请注意Add方法是有返回值的,而Print是没有返回值的,对于没有返回值的函数,Lambda表达式体必须是语句Lambda即new, call, decrement,increment,赋值等表达完整语句的操作而非表达式,这个要求和lambda是一样的。
- 属性和索引器
属性和索引器有getter和setter方法,表达式体可以用来写只读的属性和索引器,表达式体就是对应getter方法的方法体,例子如下:
public Author this[int id] => store[id]; public string FullName => firstName + lastName;
注意这里不需要给出get关键字,编译器会做隐式推断。
完整例子如下
internal class ExpressionBody { private string firstName = "first name"; private string lastName = "lastName"; private List<Author> store = new List<Author> { new Author(DateTime.Now), new Author(DateTime.Now) }; public int Add(int op1, int op2) => op1 + op2; public void Print(string message) => Console.WriteLine("Hello " + message); public Author this[int id] => store[id]; public string FullName => firstName + lastName; //public int Add(int op1, int op2) //{ // return op1 + op2; //} //public void Print(string message) //{ // Console.WriteLine("Hello " + message); //} }