一. 语法糖简介
语法糖也译为糖衣语法,是由英国计算机科学家彼得·约翰·兰达(Peter J. Landin)发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用。通常来说使用语法糖能够增加程序的可读性,从而减少程序代码出错的机会。
需要声明的是“语法糖”这个词绝非贬义词,它可以给我带来方便,是一种便捷的写法,编译器会帮我们做转换;而且可以提高开发编码的效率,在性能上也不会带来损失。
在编译器发展早期,编译器科学家门一直在想方设法的优化编译器生成的代码,这个时候,编译器做的主要是对机器优化,因为那个时候机器的时间非常宝贵,机器运算速度也不快,今天我们有了足够好的机器了(但并不是说我们可以不关注性能的编写程序),而且作为编写软件的人来说,比机器的时间宝贵得多,所以今天的编译器也在向人优化了,从编程语言的发展之路来讲,今天的编程语言比昨天的语言更高级,也更人性化了,我们只要编写更少的代码,更符合人的思维的代码,而只要关注我们值的关注的地方。体力活儿就交给编译器吧。
二. 常用语法糖
1. 自动属性
(1). 传统的方式在类中声明一个属性,需要先声明一个私有变量的字段,然后在配合公有属性,如下面的:userId属性。
(2). 利用自动属性:不需要字段,声明一个空属性,直接get,set(快捷键:prop),编译时编译器为我们生成存取数据的字段. 如下面的:userName属性。
1 public class userInfor2 {3 //私有字段4 private string _userId;5 //公有属性6 public string userId7 {8 get9 {
10 return _userId;
11 }
12 set
13 {
14 _userId = value;
15 }
16 }
17
18 public string useName { get; set; }
19
20
21 /// <summary>
22 /// 为了后面的反射调用
23 /// </summary>
24 public void Test()
25 {
26 Console.WriteLine("我是一个方法");
27 }
28
29 }
2. var和dynamic
(1). var类型:声明变量的时候可以不指定类型,由编译器编译的时候来指定类型。
①:必须在定义的时候初始化
②:必须是局部变量
③:一旦初始化完成,不能再给变量赋与初始值不同类型的值了,但是可以赋相同类型的不同值.
④:var在效率是和使用强类型方式定义变量是一样的
(2). dynamic类型:编译期间不做任何检查,运行期间才确定类型。
①:定义的时候可以不必初始化
②:可以是全局变量,也可以是局部变量
dynamic在反射中的应用:通过反射拿到类后,赋值给dynamic类型,该类型的对象可以直接点来调用方法
缺点:dynamic在运行的时候才进行检测,导致编译的时候即使有错误也不会被发现; 不能用dynamic类型给确定类型的变量进行赋值
1 public class CompareTwo2 {3 //2. 可使用的范围4 // var b1 = 1; //报错,var必须定义在方法呃逆5 // dynamic b2 = 2; //正常6 public static void Show()7 {8 {9 //1. 初始化比较
10 //var a1; //报错,定义的时候必须初始化
11 //dynamic a2; //正常
12 }
13 {
14 //3. 赋值问题
15 var c1 = 2;
16 //c1 = "2"; //报错,初始化完成不能赋值不同类型的变量
17 c1 = 3; //正常,初始化完成可以赋值相同类型的变量的不同值
18
19 dynamic d1 = 2;
20 d1 = "2"; //正常,运行时检查进行类型指定,所以在编译的时候不会报错
21 d1 = 3; //正常
22
23
24 var userInfor = new userInfor();
25 userInfor.useName = "1";
26 userInfor.userId = "2";
27 // userInfor.fk123(); //报错,编译不通过
28
29 dynamic userInfor2 = new userInfor();
30 userInfor2.userId = "1";
31 userInfor2.useName = "ypf";
32 //调用不存在的方法 (因为编译的时候根本不检查,所以不会报错,运行的时候报错)
33 //userInfor2.fk123(); //编译期间不报错,运行的时候报错
34
35 }
36 {
37 //4. dynamic在反射中的应用
38 //4.1 常规反射调用方法
39 Type type1 = typeof(userInfor);
40 object oStudent = Activator.CreateInstance(type1);
41 MethodInfo method = type1.GetMethod("Test");
42 method.Invoke(oStudent,null);
43
44 //4.2 利用dynamic简化调用
45 //定义和编译的时候可以是任何类型,运行的时候进行转换
46 dynamic oStudent2 = Activator.CreateInstance(type1);
47 oStudent2.Test();
48
49 }
50 }
51 }
3. 可选参数
给方法的参数可以指定默认值,如果在调用该方法的时候,不传入该参数,则走默认值,传入的话,则覆盖默认参数.
(1). 有默认值的参数必须定义在没有默认值的参数之后
(2). 默认参数必须是常量,不能是变量
(3). ref和out参数不能指定默认值
1 public class OptionalParas
2 {
3 public static void Test(string useName,string userPwd,int userAge=16,string userSex="男")
4 {
5 Console.WriteLine("userName:{0},userPwd:{1},userAge:{2},userSex:{3}", useName, userPwd, userAge, userSex);
6 }
7 }
1 //3.可选参数
2 Console.WriteLine("----------------------3.可选参数-------------------------");
3 OptionalParas.Test("ypf1", "123456");
4 OptionalParas.Test("ypf2", "654321", 56, "女");
4. 对象(集合)初始化器
1 public class ObjectInitilize2 {3 public static void Test()4 {5 6 //一.对象初始化7 //1. 传统初始化对象的方式8 userInfor uInfor1 = new userInfor();9 uInfor1.userId = "1";
10 uInfor1.useName = "ypf1";
11
12 //2.对象初始化器
13 userInfor uInfor2 = new userInfor()
14 {
15 userId="2",
16 useName="ypf2"
17 };
18 userInfor uInfor3 = new userInfor()
19 {
20 userId = "3",
21 useName = "ypf3"
22 };
23
24 //二. 集合初始化
25 //1. 传统方式
26 List<userInfor> uList = new List<userInfor>();
27 uList.Add(uInfor1);
28 uList.Add(uInfor2);
29
30 //2. 集合初始化器
31 List<userInfor> uList2 = new List<userInfor>(){
32 uInfor1,
33 uInfor2,
34 new userInfor(){
35 userId="4",
36 useName="ypf4"
37 }
38 };
39
40 }
41 }
5. ref和out
二者共同的使用场景,将ref或out参数传入方法中,执行完方法后,可以直接使用ref或out参数值。
(1). ref必须在使用前赋值,即传入方法前进行定义并赋值
(2). out必须在方法中赋值然后使用,在调用方法前声明两个未实例化的变量,用来传入和接收使用
举例:有两个int类型的值num1和num2,分别使用ref和out传入对应的方法中,进行值交换。
1 public class CompareRO2 {3 /// <summary>4 /// ref的使用5 /// </summary>6 public static void TestRef()7 {8 int num1 = 5;9 int num2 = 10;
10 Func1(ref num1, ref num2);
11 Console.WriteLine("num1的值为:{0},num2的值为:{1}", num1, num2);
12 }
13
14 public static void TestOut()
15 {
16 int num1;
17 int num2;
18 Func2(out num1, out num2);
19 Console.WriteLine("num1的值为:{0},num2的值为:{1}", num1, num2);
20 }
21
22 #region ref版参数值交换
23 public static void Func1(ref int n1, ref int n2)
24 {
25 int a = n1;
26 n1 = n2;
27 n2 = a;
28 }
29 #endregion
30
31 #region out版参数值交换
32 public static void Func2(out int n1,out int n2)
33 {
34 n1 = 5;
35 n2 = 10;
36 int a = n1;
37 n1 = n2;
38 n2 = a;
39 }
40 #endregion
41
42 }
6. 匿名类/匿名方法
详见:.Net进阶系列(3)-匿名类、匿名方法、扩展方法
7. 扩展方法
详见:.Net进阶系列(3)-匿名类、匿名方法、扩展方法