10.1 无参属性

10.1.1 自动实现的属性

10.1.2 合理定义属性

  • 属性可以只读或只写,而字段访问总是可读和可写的(一个例外是 readonly 字段仅在构造器中可写).
  • 属性方法可能抛出异常;字段访问永远不会.
  • 属性不能作为 out 或 ref 参数传给方法,而字段可以.
  • 属性方法可能花较长时间执行,字段访问则总是立即完成.线程同步不要使用属性,而用方法.从MashalByRefObject派生的类永远都不应该使用属性.
  • 连续多次调用,属性方法每次都可能返回不同的值,字段则每次都返回相同的值.
  • 属性方法可能造成明显的副作用(不同的赋值顺序,可能会出现不同的行为),字段访问则永远不会.
  • 属性方法可能需要额外的内存,或者返回的引用并非指向对象状态一部分,造成对返回对象的修改作用不到原始对象身上.

10.1.3 对象和集合初始化器

    String s= new Employee (){ Name="Jeff", Age=45 }.ToString().ToUpper();
//如果想调用的本来就是一个无参构造器,c#还允许省略起始大括号之前的圆括号.
String s= new Employee { Name="Jeff", Age=45 }.ToString().ToUpper();

10.1.4 匿名类型

    var o1=new{ Name="Jeff", Year=1964};
Console.WriteLine("Name={0},Year={1}",o1.Name,o1.Year); //或
String Name="Grant";
DateTime dt= DateTime.Now;
//有两个属性的一个匿名类型
//1.String Name 属性设为"Grant"
//2.Int32 Year 属性设为dt中的年份
var o2 = new { Name, dt.Year };
Console.WriteLine("Name={0},Year={1}",o2.Name,o2.Year);
  • 如果源代码中定义了多个匿名类型,且这些类型具有相同的结构,编译器只会创建一个匿名类型定义,但创建该类型的多个实例.o1=o2
  • 匿名类型的实例不能泄露到方法外部.
  • 方法原型不能接受匿名类型的参数.
  • 方法不能返回匿名类型的引用.

10.1.5 System.Tuple类型

    //Tuple没啥好写的,还是写写dynamic吧
dynamic e = new System.Dynamic.ExpandoObject();
e.x = 6; //添加一个Int32 'x'属性,其值为6
e.y = "Jeff"; //添加一个String 'y'属性,其值为"Jeff"
e.z = null; //添加一个Object 'z'属性,其值为null //查看所有属性及其值:
foreach (var v in (IDictionary<String, Object>)e)
Console.WriteLine("key={0},v={1}", v.Key, v.Value);

10.2 有参属性(c#称为索引器)

    public sealed class BitArray
{
//容纳了二进制位的私有字节数组
private byte[] m_byteArray;
private int m_numBits; //下面的构造器用于分配字节数组,并将所有位设为0
public BitArray(int numBits)
{
//先验证实参
if (numBits <= 0)
throw new ArgumentOutOfRangeException(nameof(numBits)); //保存位的个数
m_numBits = numBits;
//为位数组分配字节
m_byteArray = new byte[(numBits + 7) / 8];
} //下面是索引器(有参属性)
public bool this[int bitPos]
{
//下面是索引器的get访问器方法
get
{
//先验证实参
if ((bitPos < 0) || (bitPos >= m_numBits))
throw new ArgumentOutOfRangeException(nameof(bitPos)); //返回指定索引处的位的状态
return (m_byteArray[bitPos / 8] & (1 << (bitPos % 8))) != 0;
} //下面是索引器的set访问器方法
set
{
if ((bitPos < 0) || (bitPos >= m_numBits))
throw new ArgumentOutOfRangeException(nameof(bitPos), bitPos.ToString()); if (value)
{
//将指定索引处的位设为true
m_byteArray[bitPos / 8] = (byte)(m_byteArray[bitPos / 8] | (1 << (bitPos % 8)));
}
else
{
//将指定索引处的位设为false
m_byteArray[bitPos / 8] = (byte)(m_byteArray[bitPos / 8] & ~(1 << (bitPos % 8)));
}
}
}
}

BitArray类的索引器用起来很简单:

//分配含14个位的BitArray数组
BitArray ba = new BitArray(14); //调用set访问器方法,将编号为偶数的所有位都设为true
for (int x = 0;x < 14; x++){
ba[x]=(x % 2 == 0);
} //调用get访问器方法显示所有位的状态
for (int x = 0; x < 14; x++){
Console.WriteLine("Bit " + x + " is " +(ba[x] ? "On" : "Off"));
}
  • 可以通过IndexerNameAttribute特性改变编译器使用的索引器名称(默认为Item,并在前面加get_或set_前缀).System.String改变索引器名称为 Chars .
  • c#允许一个类型定义多个索引器,只要索引器的参数集不同.
  • 对于支持多个有参属性的编程语言,必须选中一个有参属性,通过DefaultMemberAttribute特性来标识.这是C#代码唯一能访问的有参属性.

10.3 调用属性访问器方法时的性能

  • 对于简单的get和set访问器方法,JIT编译器会将代码内联(inline,或者说嵌入).

10.4 属性访问器的可访问性

// ⑴
public class SomeType{
private string m_name;
public string Name {
get { return m_name; }
// ⑵
protected set { m_name = value; }
}
}
  • ⑴ 必须为属性本身指定限制最小的可访问性
  • ⑵ 两个访问器只能选择一个来使用限制较大的

返回目录

<NET CLR via c# 第4版>笔记 第10章 属性的更多相关文章

  1. <NET CLR via c# 第4版>笔记 第19章 可空值类型

    System.Nullable<T> 是结构. 19.1 C# 对可空值类型的支持 C# 允许用问号表示法来声明可空值类型,如: Int32? x = 5; Int32? y = null ...

  2. <NET CLR via c# 第4版>笔记 第18章 定制特性

    18.1 使用定制特性 FCL 中的几个常用定制特性. DllImport 特性应用于方法,告诉 CLR 该方法的实现位于指定 DLL 的非托管代码中. Serializable 特性应用于类型,告诉 ...

  3. <NET CLR via c# 第4版>笔记 第17章 委托

    17.1 初识委托 .net 通过委托来提供回调函数机制. 委托确保回调方法是类型安全的. 委托允许顺序调用多个方法. 17.2 用委托回调静态方法 将方法绑定到委托时,C# 和 CLR 都允许引用类 ...

  4. <NET CLR via c# 第4版>笔记 第16章 数组

    //创建一个一维数组 int[] myIntegers; //声明一个数组引用 myIntegers = new int[100]; //创建含有100个int的数组 //创建一个二维数组 doubl ...

  5. <NET CLR via c# 第4版>笔记 第13章 接口

    13.1 类和接口继承 13.2 定义接口 C#用 interface 关键字定义接口.接口中可定义方法,事件,无参属性和有参属性(C#的索引器),但不能定义任何构造器方法,也不能定义任何实例字段. ...

  6. <NET CLR via c# 第4版>笔记 第12章 泛型

    泛型优势: 源代码保护 使用泛型算法的开发人员不需要访问算法的源代码.(使用c++模板的泛型技术,算法的源代码必须提供给使用算法的用户) 类型安全 向List<DateTime>实例添加一 ...

  7. <NET CLR via c# 第4版>笔记 第5章 基元类型、引用类型和值类型

    5.1 编程语言的基元类型 c#不管在什么操作系统上运行,int始终映射到System.Int32; long始终映射到System.Int64 可以通过checked/unchecked操作符/语句 ...

  8. <NET CLR via c# 第4版>笔记 第6章 类型和成员基础

    6.1 类型的各种成员 6.2 类型的可见性 public 全部可见 internal 程序集内可见(如忽略,默认为internal) 可通过设定友元程序集,允许其它程序集访问该程序集中的所有inte ...

  9. <NET CLR via c# 第4版>笔记 第7章 常量和字段

    7.1 常量 常量 是值从不变化的符号.定义常量符号时,它的值必须能够在编译时确定. 只能定义编译器识别的基元类型的常量,如果是非基元类型,需把值设为null. 常量的值直接嵌入代码,所以不能获取常量 ...

随机推荐

  1. (2.16)Mysql之SQL基础——函数

    (2.16)Mysql之SQL基础——函数 关键词:mysql函数,mysql自定义函数,mysql聚合函数,mysql字符串函数,mysql数值函数 1.自定义函数 -- (1)一般形式 creat ...

  2. python 面向对象 公有属性 用在哪里

    公有属性也可以叫做静态字段 如果每个对象都有一个共同的值 , 应该把它设置为公有属性 公有属性使用场景,每个对象中保存相同的东西时,可以使用公有属性 类找公有属性 过程

  3. Openstack架构简介(一)

    1.1.1openstack介绍: openstack是(infrastructure as a service,基础设置即服务)IAAS架构的实现,OpenStack是一个由NASA(美国国家航空航 ...

  4. Miller-Rabin素数测试算法(POJ1811Prime Test)

    题目链接:http://poj.org/problem?id=1811 题目解析:2<=n<2^54,如果n是素数直接输出,否则求N的最小质因数. 求大整数最小质因数的算法没看懂,不打算看 ...

  5. 【PGM】Representation--Knowledge Engineering,不同的模型表示,变量的类型,structure & parameters

    Part 1. 重要的区别: Template based   vs.   specific Directed  vs.  undirected Generative  vs.  discrimina ...

  6. Android初体验之Monkey和MonkeyRunner

    原文地址https://blog.csdn.net/mad1989/article/details/38087737 Monkey 什么是Monkey Monkey是Android中的一个命令行工具, ...

  7. Linux系统——DNS

    DNS系统的作用1. DNS服务器Internet中,大部分网站.邮件服务等服务器都使用了域名形式的地址,这种地址形式要比使用IP地址形式更加直观,更加容易被用户记住.FQDN格式(完整域名格式):在 ...

  8. QML类型说明-ParallelAnimation

    ParallelAnimation ImportStatement:   import QtQuick2.2 Inherits:     Animation DetailedDescription S ...

  9. Vue学习笔记之Vue指令系统介绍

    所谓指令系统,大家可以联想咱们的cmd命令行工具,只要我输入一条正确的指令,系统就开始干活了. 在vue中,指令系统,设置一些命令之后,来操作我们的数据属性,并展示到我们的DOM上. OK,接下来我们 ...

  10. Python3.x:open()文件操作

    Python3.x:open()文件操作 open/文件操作: #open(路径+文件名,读写模式) #读写模式:r只读,r+读写,w新建(会覆盖原有文件),a追加,b二进制文件.常用模式 f=ope ...