<NET CLR via c# 第4版>笔记 第10章 属性
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章 属性的更多相关文章
- <NET CLR via c# 第4版>笔记 第19章 可空值类型
System.Nullable<T> 是结构. 19.1 C# 对可空值类型的支持 C# 允许用问号表示法来声明可空值类型,如: Int32? x = 5; Int32? y = null ...
- <NET CLR via c# 第4版>笔记 第18章 定制特性
18.1 使用定制特性 FCL 中的几个常用定制特性. DllImport 特性应用于方法,告诉 CLR 该方法的实现位于指定 DLL 的非托管代码中. Serializable 特性应用于类型,告诉 ...
- <NET CLR via c# 第4版>笔记 第17章 委托
17.1 初识委托 .net 通过委托来提供回调函数机制. 委托确保回调方法是类型安全的. 委托允许顺序调用多个方法. 17.2 用委托回调静态方法 将方法绑定到委托时,C# 和 CLR 都允许引用类 ...
- <NET CLR via c# 第4版>笔记 第16章 数组
//创建一个一维数组 int[] myIntegers; //声明一个数组引用 myIntegers = new int[100]; //创建含有100个int的数组 //创建一个二维数组 doubl ...
- <NET CLR via c# 第4版>笔记 第13章 接口
13.1 类和接口继承 13.2 定义接口 C#用 interface 关键字定义接口.接口中可定义方法,事件,无参属性和有参属性(C#的索引器),但不能定义任何构造器方法,也不能定义任何实例字段. ...
- <NET CLR via c# 第4版>笔记 第12章 泛型
泛型优势: 源代码保护 使用泛型算法的开发人员不需要访问算法的源代码.(使用c++模板的泛型技术,算法的源代码必须提供给使用算法的用户) 类型安全 向List<DateTime>实例添加一 ...
- <NET CLR via c# 第4版>笔记 第5章 基元类型、引用类型和值类型
5.1 编程语言的基元类型 c#不管在什么操作系统上运行,int始终映射到System.Int32; long始终映射到System.Int64 可以通过checked/unchecked操作符/语句 ...
- <NET CLR via c# 第4版>笔记 第6章 类型和成员基础
6.1 类型的各种成员 6.2 类型的可见性 public 全部可见 internal 程序集内可见(如忽略,默认为internal) 可通过设定友元程序集,允许其它程序集访问该程序集中的所有inte ...
- <NET CLR via c# 第4版>笔记 第7章 常量和字段
7.1 常量 常量 是值从不变化的符号.定义常量符号时,它的值必须能够在编译时确定. 只能定义编译器识别的基元类型的常量,如果是非基元类型,需把值设为null. 常量的值直接嵌入代码,所以不能获取常量 ...
随机推荐
- modelform动态显示select标签的对象范围
既根据当前登录人,动态显示对象相关的的select的选项,例如 A登录,只显示A的客户,B登录,只显示B自己的客户 先了解form的ModelChoiceField字段(这个表格没意义,就是引出参数q ...
- C# 反双曲余弦函数
反双曲余弦函数的定义是: T1 = Math.Log(t + Math.Sqrt(t * t - 1)); 1. 叉乘(cross product),也叫向量的外积.向量积.顾名思义,求下来的结果是一 ...
- RDD的基础知识
以下的这些分析都是基于spark2.1进行的 (一)什么是RDD A Resilient Distributed Dataset (RDD), the basic abstraction in Spa ...
- java基础知识 构造方法
在java里面,构造方法也就是构造函数 构造函数=构造方法;构造方法是一种特殊的方法,具有以下特点.(1)构造方法的方法名必须与类名相同.(2)构造方法没有返回类型,也不能定义为void,在方法名前面 ...
- Lyft Level 5 Challenge 2018 - Elimination Round
A. King Escape 签. #include <bits/stdc++.h> using namespace std; ], y[]; int f1(int X, int Y) { ...
- 20155307 2016-2017-2 《Java程序设计》第8周学习总结
20155307 2016-2017-2 <Java程序设计>第8周学习总结 教材学习内容总结 日志API:使用日志的起点是Logger类,要取得Logger类,必须使用Logger的静态 ...
- 使用spring cache和ehcache之前必须了解的
好长时间没写博客了,真的是没时间啊.ps:其实就是懒!!接下来几篇要写下缓存,这里主要写下ehcache与spring整合的内容,包括aop形式的缓存,基于注解的缓存,页面缓存这三方面吧.在这之前先要 ...
- P1083 借教室(差分+二分)
P1083 借教室 第一眼:线段树. 然鹅懒得写. 正解:差分+二分. 显然订单合法的上线可以二分 然后差分数组维护一下.没了. #include<iostream> #include&l ...
- 20155201 2016-2017-2 《Java程序设计》第三周学习总结
20155201 2016-2017-2 <Java程序设计>第三周学习总结 教材学习内容总结 - 第四章要点: 4.1类与对象 类定义时使用class关键词,基本模式为 class na ...
- iptable的使用
目录 iptables iptables iptables详解 2012-07-18 20:10:08 分类: LINUX 一:前言 防火墙,其实说白了讲,就是用于实现Linux下访问控制的功能的,它 ...