用智能的编译器来防错
自动实现的属性
编写由字段直接支持的简单属性,不再显得臃肿不堪
C#2允许为取值方法指定不同的访问权限
#region 8-1统计创建了多少个实例的Person类 public class Person { public string Name { get; private set; }//声明有公有取值方法的属性 public int Age { get; private set; } private static int InstanceCounter { get; set; }//声明私有的静态属性和锁 private static readonly object counterLock = new object(); public int InstanceCounterPerson(string name, int age) { Name = name; Age = age; lock (counterLock)//访问静态属性时使用锁。应避免锁定 public 类型,否则实例将超出代码的控制范围 { InstanceCounter++; } return InstanceCounter; } } #endregion #region 写一个有单个整数属性的结构 public struct Foo { public int Value { get; private set; }//所有字段在被设置之前,不能使用这些属性 public Foo(int value) : this()//显示的调用无参构造函数this(),编译器才知道所有字段都被明确的赋值了 { this.Value = value;//调用无参构造函数,改字段被设为默认值 } } #endregion隐式类型的局部变量
根据初始化值推断变量的类型,从而简化局部变量的声明
用var声明局部变量
使用隐式类型,唯一要做的就是将普通局部变量声明类型名称替换为var
编译器工作:获取初始化表达式在编译时的类型,并使变量也具有那种类型
变量是静态的类型,只是由编译器推断
隐式类型的限制
- 被声明的变量是一个局部变量,而不是一个静态字段和实例字段
- 变量在声明的同时被初始化
- 初始化表达式不是方法组也不是匿名函数
- 初始化表达式不是null
- 语句中只声明了一个变量
- 你希望变量拥有的类型是初始化编译时的类型
- 初始化表达式不包含正在声明的变量
可以这样写:var starter=(ThreadStart)delegate(){Console.WretrLine()}
同样的也适用于null
var argss = Environment.GetCommandLineArgs();//用方法的调用结果来初始化一个变量隐式类型的建议
- 如果代码让人一眼就能看出变量类型很重要,就使用显示类型
- 如果变量直接用一个构造函数初始化,而且类型名称很长,就考虑使用隐式类型
- 如果变量的确切类型不重要,而且它的本质在当前上下文已经很清楚,就用隐式类型,从而不去强调代码具体是如何达到目标的,而是关注它想要达到什么目标
- 看哪个顺眼
用一个表达式就能轻松创建初始化对象
对象初始化程序:初始化列表指定了在对象创建好后,如何对其进行初始化
#region 8-2一个相当简单的Person类 public class Person { public int Age { get; set; } public string Name { get; set; } List<Person> friends = new List<Person>();//对象创建时,以留空的方式创建,而不是保留空引用 public List<Person> Friends { get { return friends; } } Location home = new Location();//对象创建时,以留空的方式创建,而不是保留空引用 public Location Home { get { return home; } } public Person() { } public Person(string name) { Name = name; } } public class Location { public string Country { get; set; } public string Town { get; set; } } #endregion设置简单属性
对象初始化程序最常用于设置属性
Person tom1 = new Person(); tom1.Name = "Tom"; tom1.Age = 9; Person tom2 = new Person { Name = "Tom", Age = 9 };//tom1在IL中 Person tom3 = new Person("Tom"); tom3.Age = 9; Person person3 = new Person("Tom") { Age = 9 };//tom3在IL中 Person[] family = new Person[] { new Person{Name="Holly1",Age=36}, new Person{Name="Holly2",Age=36}, new Person{Name="Holly3",Age=36}, new Person{Name="Holly4",Age=36} };为嵌入对象设置属性
Person tom4 = new Person("Tom");//设置一个嵌入对象属性 tom4.Age = 9; tom4.Home.Country = "CS"; tom4.Home.Town = "WC"; Person tom5 = new Person("Tom") { Age = 9, Home = { Country = "CS", Town = "WC" } };//设置一个嵌入对象属性 Person tom6 = new Person("Tom");//设置一个嵌入对象属性 tom6.Age = 9; tom6.Home.Country = "CS"; tom6.Home.Town = "WC";集合初始化程序
#region 集合初始化程序 var names = new List<string> { "Holly","jon","Tom","Robin","William" }; List<string> names1 = new List<string>();//动态添加,IL中无该处代码 names1.Add("Holly1"); names1.Add("jon1"); names1.Add("Tom1"); names1.Add("Robin1"); names1.Add("William1"); Dictionary<string, int> nameAgeMap = new Dictionary<string, int>//任何实现IEnumerable的类型,只要它为初始化列表中出现的每个元素提供了一个公有Add方法,就可以使用这个特性 { {"Holly",36},//Add方法被调用3次,如果Add有多个重载版本,那么初始化列表每个不用元素都可以调用不通的重载版本 {"Jon",36}, {"Tom",9} }; #endregion在其他对象初始化程序中填充集合
与对象初始化程序组合使用
#region 8-3使用对象1和集合初始化程序来构建一个“富对象” Person tom = new Person { Name = "Tom", Age = 9, Home = { Town = "WC", Country = "CS" },//初始化嵌入对象 Friends = //对象初始化程序来初始化集合 { new Person{Name="Alberto"}, new Person("MAx"), new Person{Name="Zak",Age=9}, new Person("Ben"), new Person("Alice") { Age=9, Home={Town="C",Country="S"} } } }; #endregion初始化特性的应用
常量集合
设置单元测试
builder模式
隐式类型数组
MyMethod(new[] {"HE","EH"});
首先构造一个集合,其中包括大括号内所有的表达式的编译时的类型,在这个类型集合中,如果其他所有类型都能隐式转换为其中一种类型,改类型即为数组类型
首先构造一个集合,其中包括大括号内所有的表达式的编译时的类型,在这个类型集合中,如果其他所有类型都能隐式转换为其中一种类型,改类型即为数组类型
匿名类型
#region 8-4创建具有Name和Age属性的匿名类型对象 var tom = new { Name = "Tom", Age = 9 };//匿名对象初始化程序 Console.WriteLine("{0},{1}", tom.Name, tom.Age);//属性具有和初始化程序中的表达式一样的类型,值在创建匿名对象初始化程序中指定 #endregion #region 8-5用匿名类型填充数组,并计算总年龄 var family = new[]//使用隐式类型的数组初始化程序 { new{Name="Holly",Age=36},//使用匿名对象初始化程序 new{Name="Jon",Age=32}//如果生成不同的类型,编译器无法判断声明数组的类型 }; int totalAge = 0; foreach (var person in family)//对每个人使用隐式类型 { totalAge += person.Age; } Console.WriteLine("Total age:{0}", totalAge); #endregion投影初始化程序
#region 8-6从Person对象转换成一个名字和一个成年标志 List<Person> family = new List<Person> { new Person{Name="Holly",Age=36}, new Person{Name="Jon",Age=23}, new Person{Name="Jon",Age=23} }; var converted = family.ConvertAll(delegate(Person person) { return new { person.Name, IsAdult = (person.Age >= 18) }; });//投影初始化程序:如果不指定属性名称,而只指定用于求值的表达式,它就会使用表达式最后一部分作为名称————前提是它只能是一个简单的字段和属性 foreach (var person in converted) { Console.WriteLine("{0} is an adult?{1}", person.Name, person.IsAdult); } #endregion
如本文对您有帮助请移步右下角,推荐本文,谢谢大家的点赞,因为您的支持是我最大动力
浙公网安备 33010602011771号