[C#6] 5-自动属性增强

0. 目录

C#6 新增特性目录

1. 老版本代码

 1 internal class Person
 2 {
 3     public string Name { get; private set; }
 4     public int Age { get; private set; }
 5
 6     public Person(string name,int age)
 7     {
 8         Name = name;
 9         Age = age;
10     }
11 }

通常情况下,C#的属性可以很好的帮助我们完成工作,比如上面的代码。在为属性赋值的时候,我们可以在任意地方为其赋值。但是并没有一种像是字段一样的声明且立即初始化的语法来简化默认值的设定。C#6为我们带来了这种新的语法,像是为字段赋值一样为属性赋值。

我们也知道,C#的属性实际上是一个编译器自动生成的私有字段、get_xxx和set_xxx、一条元数据组成,比如上面的代码编译后:

<Name>k__BackingField字段的IL

1 .field private string ‘<Name>k__BackingField‘
2 .custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 )
3 .custom instance void [mscorlib]System.Diagnostics.DebuggerBrowsableAttribute::.ctor(valuetype [mscorlib]System.Diagnostics.DebuggerBrowsableState) = ( 01 00 00 00 00 00 00 00 )

表示一个私有字段,第2行分别表示这个自动是编译器自动生成的,第3行表示该字段不显示在Debugger窗口中。

get_Name方法的IL:

 1 .method public hidebysig specialname instance string
 2         get_Name() cil managed
 3 {
 4   .custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 )
 5   // Code size       7 (0x7)
 6   .maxstack  8
 7   IL_0000:  ldarg.0
 8   IL_0001:  ldfld      string csharp6.Person::‘<Name>k__BackingField‘
 9   IL_0006:  ret
10 } // end of method Person::get_Name

这也是一个自动生成的方法。

set_Name方法的IL:

 1 .method private hidebysig specialname instance void
 2         set_Name(string ‘value‘) cil managed
 3 {
 4   .custom instance void [mscorlib]System.Runtime.CompilerServices.CompilerGeneratedAttribute::.ctor() = ( 01 00 00 00 )
 5   // Code size       8 (0x8)
 6   .maxstack  8
 7   IL_0000:  ldarg.0
 8   IL_0001:  ldarg.1
 9   IL_0002:  stfld      string csharp6.Person::‘<Name>k__BackingField‘
10   IL_0007:  ret
11 } // end of method Person::set_Name

同样是一个自动生成的方法。

Name属性的IL:

1 .property instance string Name()
2 {
3   .get instance string csharp6.Person::get_Name()
4   .set instance void csharp6.Person::set_Name(string)
5 } // end of property Person::Name

表示Name属性由一个get方法和set方法组成。

2. 自动属性增强语法

 1 internal class Person
 2 {
 3     //声明读写属性、且初始化默认值
 4     public string Name { get; set; } = "blackheart";
 5
 6     //声明只读属性、且初始化默认值
 7     public int Age { get; } = 1;
 8
 9     //声明只读属性
10     public string Note { get; }
11
12     public Person(string note)
13     {
14         //在构造器中为只读属性初始化默认值
15         Note = note;
16     }
17
18     private void func1()
19     {
20         //error,只能在构造器中初始化
21         //Note = "123";
22         //Age = 1;
23         //可以修改,因为有set访问器
24         Name = "new name";
25     }
26 }

这种新语法会在没有set访问器的时候把隐藏的私有字段设置为只读字段(readonly ),只允许在声明的时候设置初始值或者在构造器里面赋值。看看IL:

只有Name属性具有set_Name方法,而Age和Note属性则没有set访问器,且对应的私有字段被设置为"initonly",表示这是一个只读字段。

构造器方法,Name{get;set;}="blackheart"和Age{get;}=1的初始化操作部分被转移到实例构造函数".ctor"方法中。

 1 .method public hidebysig specialname rtspecialname
 2         instance void  .ctor(string note) cil managed
 3 {
 4   // Code size       34 (0x22)
 5   .maxstack  8
 6   IL_0000:  ldarg.0
 7   IL_0001:  ldstr      "blackheart"
 8   IL_0006:  stfld      string csharp6.Person::‘<Name>k__BackingField‘
 9   IL_000b:  ldarg.0
10   IL_000c:  ldc.i4.1
11   IL_000d:  stfld      int32 csharp6.Person::‘<Age>k__BackingField‘
12   IL_0012:  ldarg.0
13   IL_0013:  call       instance void [mscorlib]System.Object::.ctor()
14   IL_0018:  nop
15   IL_0019:  nop
16   IL_001a:  ldarg.0
17   IL_001b:  ldarg.1
18   IL_001c:  stfld      string csharp6.Person::‘<Note>k__BackingField‘
19   IL_0021:  ret
20 } // end of method Person::.ctor

和之前的语法生成的代码可以说是一致的,均是生成为一个字段、get_xxx和set_xxx方法和对应的属性元数据,本质依然是编译器的语法简化。

3. 参考

C# Auto-property enhancements

时间: 2024-10-25 05:07:25

[C#6] 5-自动属性增强的相关文章

自动属性

2.0的时候我们定义属性要这么定义private int _age;public int Age{get { return _age; }set { _age = value; }}到3.0的时候 新加了自动属性可以这么定义了public int Age {get;set;}public int Age {get;private set;} 最终生成的il代码都是一样的,上面的事情编译器帮做了. CLR说,自动实现的属性(AIP),它是为了封装一个支持字段而创建的一个属性.不能用,尽量不用第一.

.NET中那些所谓的新语法之一:自动属性、隐式类型、命名参数与自动初始化器

开篇:在日常的.NET开发学习中,我们往往会接触到一些较新的语法,它们相对以前的老语法相比,做了很多的改进,简化了很多繁杂的代码格式,也大大减少了我们这些菜鸟码农的代码量.但是,在开心欢乐之余,我们也不禁地对编译器内部到底为我们做了哪些事儿而感到好奇?于是,我们就借助反编译神器,去看看编译器到底做了啥事!其实本篇中很多都不算新语法,对于很多人来说可能都是接触了很久了,这里主要是针对.NET的老版本来说,是一个“相对”的新语法. /* 新语法索引 */ 1.自动属性 Auto-Implemente

38属性的种种,只读只写属性、自动属性、静态属性、抽象属性、接口属性

□ 只读属性 public class Example {     string name;     public string Name     {         get {return name;}     } } □ 只写属性 public class Example {     string name;     public string Name     {         set {name = value;}     } } □ 可读可写属性 public class Examp

MVC4 数据验证、特性、自动属性总结

最近悟出来一个道理,在这儿分享给大家:学历代表你的过去,能力代表你的现在,学习代表你的将来. 十年河东十年河西,莫欺少年穷 学无止境,精益求精    最近在做自学MVC,遇到的问题很多,索性一点点总结下. MVC4数据验证: 本例以程序为例就行说明: public class MyModel { [Required(ErrorMessage = "请输入收件人姓名")] public string Uname { get; set; } [Required(ErrorMessage =

[读书笔记]C#学习笔记五: C#3.0自动属性,匿名属性及扩展方法

前言 这一章算是看这本书最大的收获了, Lambda表达式让人用着屡试不爽, C#3.0可谓颠覆了我们的代码编写风格. 因为Lambda所需篇幅挺大, 所以先总结C#3.0智能编译器给我们带来的诸多好处, 下一遍会单独介绍Lambda表达式. 这篇主要包括的内容有: 自动属性,隐式类型,对象集合初始化,匿名类型,扩展方法. 下面一起来看下C#3.0 所带来的变化吧. 1,自动实现的属性在C#3.0之前, 定义属性时一般会像下面这样去编写代码: 1 class Person 2 { 3 //定义私

自动属性(编译阶段和运行阶段的不同是什么,托管和非托管的不同)

public class student { public int Id{get;set;}这个就是自动属性,在编译阶段完成 } 问题1:编译阶段和运行阶段的不同 编译阶编译器干了什么事情 答: 问题2:运行阶段干了什么事情 答: 问题3:托管代码和非托管代码的区别 4.

项目知识储备-自动属性 隐式类型Var

在项目中应用到的一些基本的小知识点,也不知道原理是什么 ,也不知道为什么.今天查了一下,原来这些东西在 之前的视频中就看到过.小小总结一下. 在 C# 3.0 和更高版本中,当属性的访问器中不需要其他逻辑时,自动实现的属性可使属性声明更加简洁. 客户端代码还可通过这些属性创建对象. 知识点一:自动属性 例如下面 class myClass{ //一般属性用法 privae string name; public string Name{ get { return name;} set { nam

C#语法糖之第一篇:自动属性&amp;隐式类型

今天给大家分享一下C#语法糖的简单的两个知识点吧. 自动属性:在 C# 4.0 和更高版本中,当属性的访问器中不需要其他逻辑时,自动实现的属性可使属性声明更加简洁. 客户端代码还可通过这些属性创建对象. get and set accessors." id="mt3">如下面的示例所示声明属性时,编译器将创建一个私有的匿名支持字段,该字段只能通过属性的 get 和 set 访问器进行访问. 我们C#4.0以前的传统方式的属性是用来封装字段的,这里我简单的对比一下这两种方

c#-关于自动属性的思考

参考:c#-关于自动属性的思考 我的理解:自动属性跟 公有字段 一模一样,编程习惯而已.目前是这么认为的. 自动属性:public string Name{  get;  set } 公有字段:public stirng Name; 我感觉两个没区别....有些人用自动属性,也许是因为编程习惯吧.