C# 6.0语法糖剖析

时间:2021-08-21 08:15:52

C# 6.0语法糖剖析

2016年12月16日 16:10:27 阅读数:586
 
版权声明:本文为博主http://www.feixueteam.net原创文章,未经博主允许不得转载。 https://blog.csdn.net/u013948187/article/details/53694486
  1. 自动属性默认初始化:
    public string Name { get; set; } = "hello world";
  2. 自动只读属性默认初始化

    public string Name1 { get; } = "hello world";

    由于初始化默认值实在构造函数中赋值的,所以跟属性只读没关系

  3. 表达式为主体的函数

    1.  
      //使用方法:
    2.  
       
    3.  
      Body Get(int x, int y) => new Body(1 + x, 2 + y);
    4.  
      //编译器生成如下:
    5.  
       
    6.  
      private Program.Body Get(int x, int y)
    7.  
      {
    8.  
      return new Program.Body(1 + x, 2 + y);
    9.  
      }
    10.  
      //简化了单行方法的编写,省去写大括号的功夫。
    11.  
       
    12.  
      //同时支持没有返回值的写法:
    13.  
       
    14.  
      void OutPut(int x, int y) => Console.WriteLine("hello world");
    15.  
      //也支持异步函数的编写:
    16.  
       
    17.  
      async void OutPut(int x, int y) => await new Task(() => Console.WriteLine("hello wolrd"));
  4. Null条件运算符

    1.  
      Customer customer = new Customer();
    2.  
      string name3 = customer?.Name;
    3.  
      //等同于:
    4.  
       
    5.  
      Customer customer = new Customer();
    6.  
      if (customer1 != null)
    7.  
      {
    8.  
      string name = customer1.Name;
    9.  
      }
    10.  
      //可以和??组合起来使用:
    11.  
       
    12.  
      if (customer?.Face2()??false)
    13.  
      //还可以2个一起用:
    14.  
       
    15.  
      int? Length = customer?.Name?.Length;
    16.  
      //也可以方法调用:
    17.  
       
    18.  
      customer?.Face();
  5. 字符串格式化
    1.  
      //原来是这样
    2.  
      var s = String.Format("{0} is {1} year {{s}} old", p.Name, p.Age);
    3.  
       
    4.  
      //新的语法糖使用起来相对更轻松些:
    5.  
      var s = $"{p.Name} is {p.Age} year{{s}} old";
    6.  
       
    7.  
      //有趣的是,新格式化方式还支持任何表达式的直接赋值:
    8.  
      var s = $"{p.Name} is {p.Age} year{(p.Age == 1 ? "" : "s")} old";
  6. 索引初始化 
    var numbers = new List<string> { [7] = "seven", [9] = "nine", [13] = "thirteen" };
  7. 异常过滤器when
    1.  
      try
    2.  
      {
    3.  
      throw new ArgumentException("string error");
    4.  
      }
    5.  
      catch (ArgumentException e) when (myfilter(e))
    6.  
      {
    7.  
      Console.WriteLine(e.Message);
    8.  
      }
    9.  
       
    10.  
      static bool myfilter(ArgumentException e)
    11.  
      {
    12.  
      return false;
    13.  
      }

    When语法作用是:在进入到catch之前、验证when括号里myfilter方法返回的bool,如果返回true继续运行,false不走catch直接抛出异常。

    使用这个filter可以更好的判断一个错误是继续处理还是重新抛出去。按照以前的做法,在catch块内如需再次抛出去,需要重新throw出去,这时的错误源是捕捉后在抛的,而不是原先的,有了when语法就可以直接定位到错误源。

  8. nameof表达式
    1.  
      string name = "";
    2.  
      Console.WriteLine(nameof(name));
  9. 扩展方法
    1.  
      using static System.Linq.Enumerable; //引入类型,而不是命名空间
    2.  
      class Program
    3.  
      {
    4.  
      static void Main()
    5.  
      {
    6.  
      var range = Range(5, 17); // Ok: 不是扩展方法
    7.  
      var odd = Where(range, i => i % 2 == 1); // Error, 不在全局作用域里
    8.  
      var even = range.Where(i => i % 2 == 0); // Ok
    9.  
      }
    10.  
      }

    首先Enumerable是个静态类,里面是各种扩展方法,比如range。static的作用是把类型的静态成员一次性导入,rang虽然是静态方法,但不能导入,比如where。

    因为扩展方法虽然是一个静态方法,但是语法规定它作为一个实例方法使用(打点),所以不能在全局作用域里当静态方法用,因此var odd = Where(range, i => i % 2 == 1)是错误的。

    但是static却能把类型的扩展方法作为扩展方法本身角色的功能导入进去,所以var even = range.Where(i => i % 2 == 0)是ok的。

    这里可能稍微有点绕,lz尽量写清楚,static新用法有2个功能:

一是:把静态成员导入,但扩展方法比较特殊、排除在外。这时static是c# 6.0的新功能。

二是:等同于把扩展方法的命名空间导入,所以在集合上可以打点调用扩展方法。这是之前就有的功能,而不是把扩展方法转成单纯的静态方法导入使用。

单独给get或者set设定访问限制符 很赞
经过两次变异的委托写法 委托 - 匿名委托 - 拉姆达
delegate(int a){
Console.WriteLine(a + x);
}

using == try finally