【转】30分钟掌握 C#6

1. 只读自动属性(Read-only auto-properties)

C# 6之前我们构建只读自动属性:

  1. public string FirstName { get; private set; }
  2. 2 public string LastName { get; private set; }

原理解析:就是编译器在生成set访问器时,它的修饰符是private,由上可知所谓的只读只是针对类外部,在类内部还是可以随意修改属性值的。

C# 6中提供了真正的只读自动属性,写法如下:

  1. 1 public string FirstName { get; }
  2. 2 public string LastName { get; }

原理解析:首先编译器会生成一个readonly的私有字段而get访问器就是返回该字段的值,由上可知该只读自动属性只能在构造函数中为其赋值。

2. 自动属性初始化器(Auto-Property Initializers)

以前自动属性的赋值操作我们只能写在方法中,如构造函数:

  1. 1 public Student(string firstName, string lastName)
  2. {
  3. FirstName = firstName;
  4. LastName = lastName;
  5. }

但在C# 6中我们可以把赋值操作当作声明的一部份,如下所示:

  1. 1 public string FirstName { get; set; } = "Corleone";
  2. 2 public string LastName { get; set; } = "Mike";

备注:其实C# 6和之前的版本都一样赋值操作最终都是在方法中完成,但后者明显更简洁直观,所以这是个不错的语法糖。

3. 函数成员的表达式体(Expression-bodied function members)

C# 6中提供的一个新语法:对于只有一条语句的方法体可以简写成表达式。如下面两种情况:

1. 方法(Methods)

  1. 1 public Student Create() => new Student();

等同于:

  1. 1 public Student Create()
  2. {
  3. return new Student();
  4. }

2. 只读属性(read only properties)

  1. 1 public string FullName => string.Format("{0},{1}", FirstName, LastName);

等同于:

  1. 1 public string FullName
  2. {
  3. get
  4. {
  5. return string.Format("{0},{1}", FirstName, LastName);
  6. }
  7. }

原理解析:上面的表达式在编译后会生成最原始的方法体和访问器,值得一提的是函数表达式体跟Lambda是两个相似但不相同的东西,函数的表

达式体只能带一句话且不能包含return关键字但Lambda 能带语句块和包含关键字。

4. 使用静态(using static)

C# 6中的一个新语法:使用类型的静态成员时可以省略其类型,如下所示:

  1. 1 using static System.String; // 先导入对应成员类型
  2. 2 public bool IsNull(string str) => IsNullOrEmpty(str);

等同于:

  1. 1 public bool IsNull(string str) => string.IsNullOrEmpty(str);

总结:该语法糖的目的是使代码变得更简洁,但这个应该是区分使用场景的,如:数学计算(Math类)使用此语法糖的确能够简洁代码提高可读

性,但在某处如果导入过多的类型那么不仅不能提高阅读性反而会增加阅读难度,因为你不知道这些成员具体属于那个类型。还有若类型本身存在

同名成员使用时则会使用类型成员覆盖。

注意:使用静态这一语法糖并不适用扩展方法,因为扩展方法的设计理念就是不修改已有代码且只能在一定范围内使用,所以在特殊情况下需要将

其当作静态方法来使用,那么使用类名调用反而是比较明智的。

5. Null条件运算符(Null-conditional operators)

稍有经验的童鞋都知道在Coding过程中经常要判断变量的值是否为null,类似这种if-else的操作还不少。这使得代码看起来十分不简洁,好在C#6

中提供了解决方法:

  1. 1 var student = new Student();
  2. 2 var firstName = student?.FirstName;

等同于:

  1. 1 var student = new Student();
  2.  
  3. 3 string firstName = null;
  4. 4 if (student != null)
  5. {
  6. firstName = student.FirstName;
  7. }

使用方法:只需替换成员访问符 . 为 ?. ,若 ?. 左边为null则整个运算符的结果也为null,否则运算符的结果就等于其成员值。假如成员的类型为值

类型则整个表达式返回的类型是对应类型的可空类型,如:

  1. 1 int? age = student?.Age;

原理解析: ?. 编译后就是 if 或 三元运算符,非赋值操作(如:call)会编译成 if,赋值操作一般会编译成三元运算符。

6. 字符串插值(String Interpolation)

C# 6中提供了一种新语法来构建格式化字符串,如:

  1. 1 var fullName = $"{student.FirstName},{student.LastName}";

等同于:

  1. 1 var fullName = string.Format("{0},{1}", student.FirstName, student.LastName);

使用方法:只需在字符串前加上$符号,然后在大括号中填写表达式(字段、方法、Lambdad...)即可。

备注:

1. 字符串插值语法支持以前所有字符串格式设置(此项仅支持 .net framework,不支持 .net core 1.0.1),如:

  1. Console.WriteLine($"平均成绩:{student.GPA:F2}");

注:因为 : 总被编译器解释为表达式与字符串格式的分隔符,所以表达式中若有条件运算符则我们需要用括号来强制编译将其解析成当前语境所要

表达的意义。如:

  1. 1 Console.WriteLine($"平均成绩:{(student.GPA > 80 ? student.GPA : 0):F2}");

2. 字符串插值语法可以嵌套,如:

  1. var score = $"我的各科成绩:{ $"数学:{student.MathScores};英语:{student.EnglishScore};"}";

原理解析:$"xxx{expression1}xxx{expression2}..." 编译后就是string.Format()。

7. 异常过滤器(Exception Filters)

C# 6中的一个新功能就是异常过滤器,它可以使我们在恰当的时机来应用Catch子句,如:

  1. try
  2. {
  3. throw new WebException("Request timed out..", WebExceptionStatus.Timeout);
  4. }
  5. catch (WebException webEx) when (webEx.Status == WebExceptionStatus.Timeout)
  6. {
  7. // Exception handling
  8. }

使用方法:try-catch() when()。

总结:异常过滤器最大的亮点就是在使用恰当的情况下可以不丢失异常引发点的堆栈信息,这对程序的排错至关重要。另外它还有很多有意思的用

法,大家可以上网查下。

8. nameof表达式(nameof Expressions)

nameof 表达式的功能是获取成员名称,如抛异常:

  1. public string FullName(Student student)
  2. {
  3. if (student == null)
  4. throw new ArgumentNullException(nameof(student));
  5.  
  6. return $"{student.FirstName},{student.LastName}";
  7. }

优点:nameof 表达式它能够理解成员,当成员被重命名时nameof表达式中也重命名了,而常量字符串表示法是没有这样的优势。

缺点:nameof 表达式生成的是不完全限定名,若你需要完全限定名 nameof 就不能帮你了。

原理解析:nameof 是编译期间就确定其(成员)字符串名称的,即编译后就是常量字符串的表现形式了。

9. 在Catch和Finally中使用Await(Await in Catch and Finally blocks)

C# 5 提供的 async 和 await 使异步编程变得极为简便,但它们也有着局限性:await在catch和finally块中不能使用。但这个问题已在C# 6中得到

了解决,如:

  1. public static async Task<string> MakeRequestAndLogFailures()
  2. {
  3. await logMethodEntrance();
  4. try
  5. {
  6. // ....
  7. var responseText = await streamTask;
  8. return responseText;
  9. }
  10. catch (System.Net.Http.HttpRequestException e) when (e.Message.Contains(""))
  11. {
  12. await logError("Recovered from redirect", e);
  13. return "Site Moved";
  14. }
  15. finally
  16. {
  17. await logMethodExit();
    }
  18. }

10. Index初始化器(Index Initializers)

这个功能并没有什么新意,其实以前就支持集合/字典 初始化器了,如:

  1. var list = new List<string>()
  2. {
  3. "Mike",
  4. "Jim"
  5. };
  6.  
  7. var dic = new Dictionary<int, string>()
  8. {
  9. { , "Mike" },
  10. { , "Jim" }
  11. };

在C# 6中只是字典初始化器支持了新的写法,如:

  1. var dic = new Dictionary<int, string>()
  2. {
  3. [] = "Mike",
  4. [] = "Jim"
  5. };

总结:暂无发现特殊的用法。

11. 改进的重载解析—编译器(Improved overload resolution)

这算不上是新语法,因为仅仅是编译器的改进,之所以一提是想让大家知道有这么一回事。以前的编译器是识别不了 Task.Run(Func<Task>())

的,如下:

  1. static Task DoThings()
  2. {
  3. return Task.FromResult();
  4. }
  5.  
  6. Task.Run(DoThings); // 此处省略方法代码...

上述代码在老版本编译器下是编译不通过的,而在新版本编译器是能编译通过的。

备注:值得一提的是新版本编译器也只是识别了Task.Run(Func<Task>()),Task.Run(Action) 还是识别不了,总的来说此功能对我们用处不大,

还不如乖乖的写回Lambda表达式。

由于篇幅所限,C# 7 新增的功能就放在下篇介绍......

【转】30分钟掌握 C#6的更多相关文章

  1. 30分钟学会XAML

    1.狂妄的WPF 相对传统的Windows图形编程,需要做很多复杂的工作,引用许多不同的API.例如:WinForm(带控件表单).GDI+(2D图形).DirectX API(3D图形)以及流媒体和 ...

  2. Shell脚本编程30分钟入门

    Shell脚本编程30分钟入门 转载地址: Shell脚本编程30分钟入门 什么是Shell脚本 示例 看个例子吧: #!/bin/sh cd ~ mkdir shell_tut cd shell_t ...

  3. JS组件系列——又一款MVVM组件:Vue(一:30分钟搞定前端增删改查)

    前言:关于Vue框架,好几个月之前就听说过,了解一项新技术之后,总是处于观望状态,一直在犹豫要不要系统学习下.正好最近有点空,就去官网了解了下,看上去还不错的一个组件,就抽空研究了下.最近园子里vue ...

  4. 2016windows(10) wamp 最简单30分钟thrift入门使用讲解,实现php作为服务器和客户端的hello world

    2016最简单windows(10) wamp 30分钟thrift入门使用讲解,实现php作为服务器和客户端的hello world thrift是什么 最简单解释 thrift是用来帮助各个编程语 ...

  5. 30分钟全面解析-SQL事务+隔离级别+阻塞+死锁

    以前总是追求新东西,发现基础才是最重要的,今年主要的目标是精通SQL查询和SQL性能优化.  本系列主要是针对T-SQL的总结. [T-SQL基础]01.单表查询-几道sql查询题 [T-SQL基础] ...

  6. 30分钟?不需要,轻松读懂IL

    先说说学IL有什么用,有人可能觉得这玩意平常写代码又用不上,学了有个卵用.到底有没有卵用呢,暂且也不说什么学了可以看看一些语法糖的实现,或对.net理解更深一点这些虚头巴脑的东西.最重要的理由就是一个 ...

  7. 【grunt第二弹】30分钟学会使用grunt打包前端代码(02)

    前言 上一篇博客,我们简单的介绍了grunt的使用,一些基础点没能覆盖,我们今天有必要看看一些基础知识 [grunt第一弹]30分钟学会使用grunt打包前端代码 配置任务/grunt.initCon ...

  8. Objective-C 30分钟入门教程

    Objective-C 30分钟入门教程 我第一次看OC觉得这个语言的语法有些怪异,为什么充满了@符号,[]符号,函数调用没有()这个,但是面向对象的高级语言也不外乎类,接口,多态,封装,继承等概念. ...

  9. AngularJS 30分钟快速入门【译】

    引用自:http://www.revillweb.com/tutorials/angularjs-in-30-minutes-angularjs-tutorial/,翻译如下: 简介 我三年前开始使用 ...

  10. 30分钟学会如何使用Shiro

    本篇内容大多总结自张开涛的<跟我学Shiro>原文地址:http://jinnianshilongnian.iteye.com/blog/2018936 我并没有全部看完,只是选择了一部分 ...

随机推荐

  1. oracle动态磁盘管理

    一.ASM发展: oracle 想做硬件: 收购sum公司的小型机业务 推出一体机Exadata 1.sum服务器 2.磁盘柜(没做存储) 推出asm建库方式配合一体机(自动使用磁盘柜的盘)(抛弃ra ...

  2. SpringCloud学习笔记(14)----Spring Cloud Netflix之Hystrix对Feign的支持

    1. Hystrix对Feign的支持 添加Feign中IUserBiz的实现类HystrixFallBack: package com.wangx.cloud.springcloud02consum ...

  3. vue循环遍历给div添加id

    html部分 <div class="img-preview" v-for="(img,i) of list" :key="img.imageK ...

  4. vue总线bus传值的一些问题

    动态组件中用总线Bus的坑 在我们的项目总难免会遇到用动态组件,这里就拿vue官方的例子为例,我们欲在组件中添加总线bus(其实官方推荐的vuex更好用,但是有时候我们只需要传一个小状态,不需要用vu ...

  5. Oralce 视图 view

    Oracle视图 Oracle的数据库对象分为五种:表,视图,序列,索引和同义词. 视图是基于一个表或多个表或视图的逻辑表,本身不包含数据,通过它可以对表里面的数据进行查询和修改.视图基于的表称为基表 ...

  6. Mysql学习总结(27)——Mysql数据库字符串函数

    注:sql的移植性比较强,函数的移植性不强,一般为数据库软件特有,例如mysql有mysql的函数,oracle有oracle的函数. 1.concat连接字符串: 从上图中可以看出,直接使用sele ...

  7. 洛谷 P2209 [USACO13OPEN]燃油经济性Fuel Economy

    P2209 [USACO13OPEN]燃油经济性Fuel Economy 题目描述 Farmer John has decided to take a cross-country vacation. ...

  8. leetCode解题报告5道题(七)

    题目一:Interleaving String Given s1, s2, s3, find whether s3 is formed by the interleaving of s1 and s2 ...

  9. Android Studio生成apk

    1.菜单Build->Generate Signed APK 2.生成android.keystore,能够依据弹框去Create new一个,也可使用命令来生成android.keystore ...

  10. Android面试精华

    SIM卡的EF文件有什么作用? SIM卡里的全部文件按树来组织: 主文件MF(Master File)--主文件仅仅有文件头,里面存放着整个SIM卡的控制和管理信息 专用文件DF(Dedicated ...