看看C# 6.0中那些语法糖都干了些什么(终结篇)

  终于写到终结篇了,整个人像在梦游一样,说完这一篇我得继续写我的js系列啦。

一:带索引的对象初始化器

  还是按照江湖老规矩,先扒开看看到底是个什么玩意。

1         static void Main(string[] args)
2         {
3             Dictionary<string, string> dic = new Dictionary<string, string>()
4             {
5                 ["Name"] = "ctrip",
6                 ["Age"] = "15"
7             };
8         }

第一眼看到这个还是蛮新鲜的,不过转眼就能想到是不是跟数组初始化器,对象初始化器一个样?你要是这么想就对了,下面我们来看看这玩意会生成

什么样的IL。

从上图中可以清楚的看到set_Item方法,这个方法就是编译器上层的索引器语法糖,就是忽悠我们提高开发效率的,不过也还行,起码让我少输入了

两个dic,然后把代码还原如下:

1             Dictionary<string, string> dic = new Dictionary<string, string>();
2             dic["Name"] = "ctrip";
3             dic["Age"] = "20";

索性趁热打铁,看看这个索引器方法的内部代码是什么样的,从下图中可以看到是一个Insert操作。

二:无参数的结构体构造函数

  不知道有多少人知道值类型在C#6.0之前是绝对不可以定义默认构造函数的,为什么这么说呢?道理很简单,因为值类型和引用类型的机制不一样,

值类型不需要new就可以在栈中分配空间,比如下面的结构体Point,只要我们定义了,就可以方便的使用point.X值。

那问题来了,如果我定义了一个默认的构造函数,并且在里面写下x=5,y=5,那谁可以告诉我,当我定义point的时候,有没有调用构造函数呢???

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Threading.Tasks;
 6 using System.Console;
 7
 8 namespace ConsoleApplication3
 9 {
10     class Program
11     {
12         static void Main(string[] args)
13         {
14             Test t = new Test();
15             var s = t.point.X;    //请问有没有调用构造函数啊?啊啊啊啊啊。。。。。。。。。。。。。。。
16             Console.Read();
17         }
18     }
19
20     public class Test
21     {
22         public Point point;
23     }
24
25     public struct Point
26     {
27         public int X;
28
29         public int Y;
30
31         public Point()
32         {
33             X = 5;
34             Y = 5;
35         }
36     }
37 }

如果执行了默认构造函数,那我point.X的时候会输出5,是不是觉得有点奇怪呢?所以基于这个原因,C#6.0之前为了避嫌,就禁止了这种默认的值

类型构造形式。但是这次在C#6.0中居然放开了,所以我就很迫不及待的去看一看到底调没调用默认构造函数,如下图:

从图中看到并没有调用默认构造函数,到这里我也知道了,只有在我new的时候才会调用,所以我就发现,值类型是在模仿引用类型的使用方式了,

个人感觉真的没有必要放开这个限制。

三:异常筛选器

  C#6.0中这个异常筛选器还真是个比较新奇的东西,不看不知道,一看吓一跳,比如下面的代码。

 1         public void Run()
 2         {
 3             try
 4             {
 5             }
 6             catch (Exception ex)
 7             if (ex.Message.Contains("timeout"))
 8             {
 9                 throw;
10             }
11         }

如果你仔细看的话,好像就是一个catch中省略了{}而已嘛?并没有看到什么其他特殊的东西,然后我就非常好奇的把上面的代码恢复到6.0版本之前,

代码如下:

 1         public void Run1()
 2         {
 3             try
 4             {
 5
 6             }
 7             catch (Exception ex)
 8             {
 9                 if (ex.Message.Contains("timeout"))
10                 {
11                     throw;
12                 }
13             }
14         }

接来下,我们就来看看这两份代码的IL到底会是个什么样子?内心狂鸡冻啊,啊啊啊啊啊啊啊。。。。都痉挛了。。。。。

可以看到,上面两份貌似相同的代码,其实生成的IL还是有很大区别的,新版代码中会用isinst判断是否为Exception的实例,并且用brtrue来判断当前是否

为null,如果是null,则不会执行ex.Message.Contains("timeout")语句了。但是老版代码并没有true/false判断,还是按照常规执行,所以现在可以知道,

其实并不是简单的省略了个"{}"大括号,这个语法糖在底层还是有些智能判断的。

好了,所有的C#6.0的语法糖分析到这里就结束了,感谢大家的关注。

时间: 2024-10-14 06:41:16

看看C# 6.0中那些语法糖都干了些什么(终结篇)的相关文章

看看C# 6.0中那些语法糖都干了些什么(中篇)

接着上篇继续扯,其实语法糖也不是什么坏事,第一个就是吃不吃随你,第二个就是最好要知道这些糖在底层都做了些什么,不过有一点 叫眼见为实,这样才能安心的使用,一口气上五楼,不费劲. 一:字符串嵌入值 我想String.Format方法就是化成灰大家都认识,比如下面代码: 1 class Bird 2 { 3 private string Name = "swallow"; 4 5 public void Fly() 6 { 7 var result = string.Format(&quo

看看C# 6.0中那些语法糖都干了些什么(上篇)

今天没事,就下了个vs2015 preview,前段时间园子里面也在热炒这些新的语法糖,这里我们就来看看到底都会生成些什么样的IL? 一:自动初始化属性 确实这个比之前的版本简化了一下,不过你肯定很好奇,到底编译器给我们做了哪些东西呢? 1 class Student 2 { 3 public string Name { get; set; } = "ctrip"; 4 } 从这张图中可以看到,在ctor中<Name>k__backingfield=“ctrip“的赋值在b

js中的new关键字都干了些什么?

new 操作符 在有上面的基础概念的介绍之后,在加上new操作符,我们就能完成传统面向对象的class + new的方式创建对象,在javascript中,我们将这类方式成为Pseudoclassical.基于上面的例子,我们执行如下代码   var obj = new Base(); 这样代码的结果是什么,我们在Javascript引擎中看到的对象模型是: new操作符具体干了什么呢?其实很简单,就干了三件事情. var obj  = {}; obj.__proto__ = Base.prot

JVM(二):Java中的语法糖

JVM(二):Java中的语法糖 上文讲到在语义分析中会对Java中的语法糖进行解糖操作,因此本文就主要讲述一下Java中有哪些语法糖,每个语法糖在解糖过后的原始代码,以及这些语法糖背后的逻辑. 语法糖 语法糖(Syntactic sugar),也译为糖衣语法,是由英国计算机科学家彼得·约翰·兰达(Peter J. Landin)发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用.通常来说使用语法糖能够增加程序的可读性,从而减少程序代码出错的机会.

【Java基础】Java中的语法糖

目录 Java中的语法糖 switch对String和枚举类的支持 对泛型的支持 包装类型的自动装箱和拆箱 变长方法参数 枚举 内部类 条件编译 断言 数值字面量 for-each try-with-resource Lambda表达式 Lambda表达式的语法 基本的Lambda例子(实现功能接口) 使用Lambdas排序集合 使用Lambdas和Streams 字符串对+号的支持 参考 语法糖(Syntactic Sugar),也称糖衣语法,指在计算机语言中添加的某种语法,这种语法对语言本身

C# 7.0中可能出现的语法

今天在MSDN上看到的微软关于微软关于C# 7.0特性的Work List,主要特性如下: Tuple增强 Tuple的可读性一直不是很好, 很多时候宁愿新写一个类也不使用Item1, Item2这种不知何物的属性: ????public Tuple<int, int> Tally(IEnumerable<int> values) { ... }????var t = Tally(myValues);????Console.WriteLine($"Sum: {t.Item

Java 中的语法糖

定义:指的是,在计算机语言中添加某种语法,这种语法能使程序员更方便的使用语言开发程序,同时增强程序代码的可读性,避免出错的机会:但是这种语法对语言的功能并没有影响.Java中的泛型,变长参数,自动拆箱/装箱,条件编译等都是 泛型:ava的泛型只在源代码存在,只供编辑器检查使用,编译后的字节码文件已擦除了泛型类型 public static void main(String[] args) { List<String> stringList = new ArrayList<String&g

vue项目中快捷语法糖

1.Vue.js是渐进式框架,采用自底向上增量开发的设计基于MVVM思想. 2.Vue 完全有能力驱动采用单文件组件和Vue生态系统支持的库开发的复杂单页应用. 3.Vue.js 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件. 4.指令带有前缀 v-,以表示它们是 Vue.js 提供的特殊属性. 5.站点上动态渲染的任意 HTML 可能会非常危险,因为它很容易导致 XSS 攻击. 6.请只对可信内容使用 HTML 插值,绝不要对用户提供的内容插值. 7.当一个 ViewM

C# 中的语法糖

1.   using 代替了 try-catch-finally 因为之前是学 Java 的,在连接数据库或者进行文件读写操作时很自然的就使用了 try-catch-finally-,在 C# 中这样写也肯定可以但是还有一种更优雅的写法(我看到别人这样说),那就是使用 using 关键字. 在使用各种流或者连接时[文件读写流,数据库连接等等],在使用完毕后,需要关闭流和连接,(这里不讨论数据库连接需要归还到池子) 之前的代码是这样的: public void Test1() { SqlConne