从反编译的角度去观察C#6.0
1. 自动属性初始化 (Initializers for auto-properties)
1.1 C#6.0 之前的写法
public class FirstExperience
{
private string _stringAutoProp = "自动初始化属性";
public string StringAutoProp
{
get{ return _stringAutoProp; }
set{ _stringAutoProp = value; }
}
}
1.2 C#6.0 中的写法
public class FirstExperience
{
public string StringAutoProp { get; set; } = "自动初始化属性";
}
1.3 反编译效果
从反编译中我们发现多了个私有字段this.<StringAutoProp>k__BackingField
;并在默认的构造函数中将“自动初始化属性”赋值给了这个字段,那么可以推断出IL中肯定会有这个匿名字段的定义,让我们再来看看IL中的体现:
用上面的两张图我们可以得出以下结论
其实自动初始化属性就是在CLR中体现就是创建私有的匿名字段,然后在构造函数中为这个匿名属性赋值,而属性的get
和Set
方法其实就是在操作这个匿名属性。
2. 只读属性的初始化(Getter-only auto-properties)
2.1 C#6.0 之前的写法
public FirstExperience(int id)
{
_id = id;
}
private readonly int _id;
public int ID
{
get { return _id; }
}
2.2 C#6.0 中的写法
public int ReadonlyProp { get; }
public FirstExperience(int _readonlyProp)
{
ReadonlyProp = _readonlyProp;
}
2.3 反编译效果
从反编译的结果我们看到构造函数中将传进来的_readonlyProp
赋值给this.<ReadonlyProp>k__BackingField
字段,而属性的get
方法操作的就是这个由编译器生成的字段,而在ReadonlyProp
属性的get
方法中打了CompilerGenerated
这个特性标签,这个标签其实就是区分编译器生成的元素与用户生成的元素,关于这个特性请移步到此。由此可以推断IL中所做的操作与自动初始化属性一致
3. 用Lambda作为函数体 (Expression bodies on method-like members)
3.1 C# 6.0 之前的写法
public override string ToString()
{
return string.Format("{0}",StringAutoProp);
}
3.2 C# 6.0 写法
public override string ToString() => string.Format("{0}", StringAutoProp);
在6.0 语法中我们看到可以使用lambda表达式直接作为函数体,那么编译器到底为我们做了什么呢?
3.3 反编译效果
从反编译中我们可以看出编译器编译的时候把lambda
函数直接编译成方法体,那么再看看IL的区别了。
从上面两张图中可以看出除了标注堆栈标注刻度
不同以外,看不出有什么差别。我也不是太明白这个地方,求园内的大牛解释解释!
4. 空值判断 (Null-conditional operators)
4.1 恶心的写法
if(xxx !=null)
{
xxx.ToString();
}
这种写法相信有非常多的朋友用过,经常为了一个是否为空的问题搞得代码非常难看,
举个栗子:我们需要获取集合或者是字符喘的长度
if(parem!=null)
{
return parem.Length;
}
return null;
这样的写法实在太恶心了,在6.0语法中我们可以这么写
public int? GetListCount(List<int> list)
{
return list?.Count();
}
4.2 实际应用的栗子
在实际应用中,经常会用到委托,而在之前我们需要调用委托时则需要判断委托是否为空如下代码:
public void Test(TestNew test)
{
if (test != null)
{
test.Invoke();
}
}
对于强迫症的我来说这样太恶心了,于是乎利用6.0的语法开始改造
public void Test1(TestNew test)
{
test?.Invoke();
}
4.3 根据上面的栗子进行反编译
从上图反编译效果可以看到两个方法基本没有差别,接下来看看IL中有什么差别:
上图为6.0语法委托调用的IL图,发现核心还是brtrue.s
做IF判断决定执行流
5. nameof表达式 (nameof expressions)
5.1 在实际项目的方法中我们经常需要这样去做参数判断
public void MyMethod(string name)
{
if (string.IsNullOrEmpty(name))
{
throw new ArgumentNullException("name");
}
}
上面代码中的name
是我们手写的字符串,在给参数name
改名时经常会忘记改下面的字符串name
,然而 6.0 解决了这个问题:
public void MyMethod(string name)
{
if (string.IsNullOrEmpty(name))
{
throw new ArgumentNullException(nameof(name));
}
}
5.2 nameof的反编译
上图中nameof(name)
被替换成了字符串name
,就如同常量一样。继续来看看IL做了什么:
从IL中我们可以看到它是直接ldstr
,既不是反射也不是拿变量的内存值,在实际运用中有时候我们会通过Type
来获取他的类名或者是直接写死的但是这样通过反编译后会发现Type.Name
在IL中其实是通过反射来获取,而直接写死的方式,在IL中其实就是定义一个变量到堆栈中然后再引用进去。
而nameof(name)
具有这两种的的有点,灵活而且相对于type
和写死的方式性能更高了。
6. 小结
其实6.0中还有很多新的语法糖,在这里就不一一介绍了,等不忙了,我在把其他的新的语法糖写出来
非常感谢您花时间读完这篇文章,如果您觉得此文不错,请点一下“推荐”按钮,您的“推荐”就是对我最大的鼓励以及不懈努力的肯定。
本文版权归作者和博客园所有,来源网址:http://www.cnblogs.com/Wesley-Zen/欢迎各位转载,转载文章之后必须在文章页面明显位置给出作者和原文连接,否则保留追究法律责任的权利以及小小的鄙视。
从反编译的角度去观察C#6.0的更多相关文章
- Android反编译基础(apktoos)--广工图书馆APK
更多精彩内容 :http://www.chenchuangfeng.com QQ:375061590 ------------------------------------------------- ...
- Android apk反编译基础(apktoos)图文教程
本文主要介绍了Android apk反编译基础,使用的工具是apktoos,我们将用图文的方式说明apktoos工具的使用方式,你可以参考这个方法反编译其它APK试试看了 很久有写过一个广工图书馆主页 ...
- Unity3D避免代码被反编译
1.Unity编译后最终会将代码编译在dll里面,无论是ios还是Android解开包以后都可以拿到dll,路径在Data/Managed/Assembly-CSharp.dll 2.IOS其实不用做 ...
- Java 反编译工具哪家强?对比分析瞧一瞧
前言 Java 反编译,一听可能觉得高深莫测,其实反编译并不是什么特别高级的操作,Java 对于 Class 字节码文件的生成有着严格的要求,如果你非常熟悉 Java 虚拟机规范,了解 Class 字 ...
- android app反编译
最新在反编译别人的apk,有时候看到别人的app有很炫的效果,就想知道别人是怎么样实现的,顺便自己参考,借鉴一下,下一次如果工作中碰到这个问题就知道怎么实现了.最近恰好放假三天,就研究了一下,一般别人 ...
- Eclipse反编译插件: Jodeclipse与JadClipse
Eclipse反编译插件: Jodeclipse与JadClipse Jodeclipse 是Jode的Eclipse插件,JadClipse是Jad的Eclipse插件,它们都是非常好的反编译插件. ...
- Eclipse安装反编译工具JadClipse
记录一下安装Eclipse的反编译插件JadClipse的过程: 1.首先准备: 1)JadClipse插件,从网站http://sourceforge.net/projects/jadclipse/ ...
- Java中String的intern方法,javap&cfr.jar反编译,javap反编译后二进制指令代码详解,Java8常量池的位置
一个例子 public class TestString{ public static void main(String[] args){ String a = "a"; Stri ...
- Android N 7 【 classes.dex】反编译失败:com.googlecode.d2j.DexException: not support version.
Microsoft Windows [版本 6.1.7601]版权所有 (c) 2009 Microsoft Corporation.保留所有权利. D:\Android反编译工具[全]\2016\d ...
随机推荐
- less样式表
LESS是一种由Alexis Sellier设计的动态层叠样式表语言.受Sass所影响,同时也影响了Sass的新语法:SCSS.[2] LESS是开源的,其第一个版本由Ruby写成,但在后续的版本当 ...
- Swift中的协议
协议: 1.Swift协议用于定义多个类型应该遵守的规范 2.协议定义了一种规范, 不提供任何实现 3.协议统一了属性名, 方法, 下标, 但是协议并不提供任何实现 4.语法格式: [修饰符] pro ...
- poj1850
是因为我好久不刷题了吗,这个题竟然做了俩小时,好几个思路都被推翻 用dp数组预处理出范围是a->a+x字符y长度有多少种递增串 然后例如def首先求a__有多少种情况那么自然后面就是只有b即dp ...
- 你好,C++(19)“老师,我这次四级考试过了没有?”——4.2 条件选择语句
4.2 条件选择语句 “老师,我这次四级考试过了没有?” 如果老师被问到这个问题,他会如何回答?是的,他会根据不同的条件选择不同的回答: 如果考试成绩大于等于60,那就回答:“恭喜你,你通过了这次考 ...
- php测试题整理(0519)
1.B/S架构和C/S架构: B/S架构是依托于浏览器的网络系统,C/S架构是基于客户端的. B/S架构: 随着Internet和WWW的流行,以往的主机/终端和C/S都无法满足当前的全球网络开放.互 ...
- Spring ioc 原理
java程序员都知道:java程序中的每个业务逻辑至少需要两个或以上的对象来协作完成,通常,每个对象在使用他的合作对象时,自己均要使用像new object() 这样的语法来完成合作对象的申请工作.你 ...
- 蚁群算法matlab实现
大家好,我是小鸭酱,博客地址为:http://www.cnblogs.com/xiaoyajiang 以下用matlab实现蚁群算法: %蚂蚁算法test %用产生的一个圆上的十个点来检验蚂蚁 ...
- caffe绘制训练过程的loss和accuracy曲线
转自:http://blog.csdn.net/u013078356/article/details/51154847 在caffe的训练过程中,大家难免想图形化自己的训练数据,以便更好的展示结果.如 ...
- NGINX小技巧--将所有目录和目录下所有文件分别给与不同的权限
为了安全,有时要将文件的权限进行限制,但,目录如果没有755,则不能进入. 所以需要分别给权限 find ./ -type f -name "*" |xargs ls -l
- Powershell ForEach-Object 循环
Powershell管道就像流水线,对于数据的处理是一个环节接着一个环节,如果你想在某一环节对流进来的数据逐个细致化的处理,可是使用ForEach-Object,$_ 代表当前的数据. 对管道对象逐个 ...