对一段c#编写的代码,有一些疑问,想通过IL中间语言看看,编译后是怎么处理的。代码如下:

         static StringBuilder sb = new StringBuilder();

         static int filecount = ;
static int dirCount = ; /// <summary>
/// 获取目录path下所有子文件名
/// </summary>
public static List<string> getAllFiles(String path, int depth = )
{
List<string> strs = new List<string>();
if (System.IO.Directory.Exists(path))
{
//所有子文件名
string[] files = System.IO.Directory.GetFiles(path);
foreach (string file in files)
{
sb.Append(new string('*', depth * ) + file + "\r\n");
strs.Add(file);
filecount++;
}
//所有子目录名
string[] Dirs = System.IO.Directory.GetDirectories(path);
foreach (string dir in Dirs)
{
dirCount++;
sb.Append(new string('*', depth * ) + dir + "\r\n");
var tmp = getAllFiles(dir, depth + ); //子目录下所有子文件名
if (!tmp.Equals(""))
{
strs.AddRange(tmp);
}
}
}
return strs;
}

这段代码的功能是很简单的:给定一个文件夹,返回下面的所有文件(递归遍历)。我的疑问:在第11行,递归调用的时候,strs变量(用来存放所有文件的名称列表)能够保存到所有文件名吗?程序运行的结果告诉我,代码没有任何问题。代码是别人写的,如果是我写的话,可能会在方法之外定义一个变量,在递归的时候,遇到文件的时候,把名字里存进去,而递归方法就不用返回值。当然按照依赖关系来看的话,方法内部用的变量,要么是内部自定义的,要么是参数传进来的,这样对外没有依赖,方法的复用性高。毋庸置疑,代码没啥问题。第一行的sb是我加进去的。通过查看IL代码,明白了一些道理,看到了编译器为我们程序处理的一些东西,比较感兴趣,就把这段IL摘下来:

.method public hidebysig static class [mscorlib]System.Collections.Generic.List`<string>
getAllFiles(string path,
[opt] int32 depth) cil managed
{
.param [] = int32(0x00000000)
// 代码大小 262 (0x106)
.maxstack
.locals init ([] class [mscorlib]System.Collections.Generic.List`<string> strs,
[] string[] files,
[] string file,
[] string[] Dirs,
[] string dir,
[] class [mscorlib]System.Collections.Generic.List`<string> tmp,
[] class [mscorlib]System.Collections.Generic.List`<string> CS$$,
[] bool CS$$,
[] string[] CS$$,
[] int32 CS$$) // 带$符号的变量是编译器定义的临时变量
IL_0000: nop
IL_0001: newobj instance void class [mscorlib]System.Collections.Generic.List`<string>::.ctor()
IL_0006: stloc.0 //把上面实例化好的对象保存到第一个变量中,也就是strs
IL_0007: ldarg.0 //加载第一个参数,也就是path
IL_0008: call bool [mscorlib]System.IO.Directory::Exists(string)
IL_000d: ldc.i4.0 //在堆栈中加载一个int类型,值为0的常量
IL_000e: ceq
IL_0010: stloc.s CS$$0001 //把 call的结果和0作比较,把比较的结果存到这个变量中
IL_0012: ldloc.s CS$$0001 //加载这个变量到堆栈中
IL_0014: brtrue IL_00fe //如果call的结果和0相等就跳转了,也就是说001中存了true
IL_0019: nop
IL_001a: ldarg.0
IL_001b: call string[] [mscorlib]System.IO.Directory::GetFiles(string)
IL_0020: stloc.1 //存储到第一个变量:files
IL_0021: nop
IL_0022: ldloc.1 //加载到第一个变量到堆栈里
IL_0023: stloc.s CS$$0002 //把这个变量的值存到002变量中
IL_0025: ldc.i4.0
IL_0026: stloc.s CS$$0003 //把0存储到003的变量中
IL_0028: br.s IL_006c //跳转到006c行
IL_002a: ldloc.s CS$$0002 //加载数组
IL_002c: ldloc.s CS$$0003 //加载003,此时值为0
IL_002e: ldelem.ref //取数组中的一个元素
IL_002f: stloc.2 //存到变量2 file中
IL_0030: nop
IL_0031: ldsfld class [mscorlib]System.Text.StringBuilder Client.Program::sb //加载sb字段
IL_0036: ldc.i4.s 42 //加载常量42
IL_0038: ldarg.1 //加载第二个参数depth
IL_0039: ldc.i4.3 //加载常量3
IL_003a: mul //3*depth 乘法
IL_003b: newobj instance void [mscorlib]System.String::.ctor(char,
int32)
IL_0040: ldloc.
IL_0041: ldstr "\r\n"
IL_0046: call string [mscorlib]System.String::Concat(string,
string,
string) //实例化了string的实例,并把前面3个字符串给拼接起来
IL_004b: callvirt instance class [mscorlib]System.Text.StringBuilder [mscorlib]System.Text.StringBuilder::Append(string)
IL_0050: pop
IL_0051: ldloc.0 //加载第一个变量strs
IL_0052: ldloc.2 //加载第三个变量file
IL_0053: callvirt instance void class [mscorlib]System.Collections.Generic.List`<string>::Add(!)
IL_0058: nop
IL_0059: ldsfld int32 Client.Program::filecount
IL_005e: ldc.i4.
IL_005f: add //filecount加1
IL_0060: stsfld int32 Client.Program::filecount
IL_0065: nop
IL_0066: ldloc.s CS$$0003 加载了003,值为0
IL_0068: ldc.i4.
IL_0069: add
IL_006a: stloc.s CS$$0003 给003加了1,此时003的值为1,这么看来003就是个计数器,用来控制循环
IL_006c: ldloc.s CS$$0003 //加载003,第一次,变量中为0
IL_006e: ldloc.s CS$$0002 //加载002,此时为files数组
IL_0070: ldlen //取数组长度,并转为int类型
IL_0071: conv.i4
IL_0072: clt
IL_0074: stloc.s CS$$0001 //看003是否比数组长度小,把比较结果存到001中
IL_0076: ldloc.s CS$$
IL_0078: brtrue.s IL_002a //如果为true,跳转到002a行
IL_007a: ldarg.
IL_007b: call string[] [mscorlib]System.IO.Directory::GetDirectories(string)
IL_0080: stloc.
IL_0081: nop
IL_0082: ldloc.
IL_0083: stloc.s CS$$
IL_0085: ldc.i4.
IL_0086: stloc.s CS$$
IL_0088: br.s IL_00ef
IL_008a: ldloc.s CS$$
IL_008c: ldloc.s CS$$
IL_008e: ldelem.ref
IL_008f: stloc.s dir
IL_0091: nop
IL_0092: ldsfld int32 Client.Program::dirCount
IL_0097: ldc.i4.
IL_0098: add
IL_0099: stsfld int32 Client.Program::dirCount
IL_009e: ldsfld class [mscorlib]System.Text.StringBuilder Client.Program::sb
IL_00a3: ldc.i4.s
IL_00a5: ldarg.
IL_00a6: ldc.i4.
IL_00a7: mul
IL_00a8: newobj instance void [mscorlib]System.String::.ctor(char,
int32)
IL_00ad: ldloc.s dir
IL_00af: ldstr "\r\n"
IL_00b4: call string [mscorlib]System.String::Concat(string,
string,
string)
IL_00b9: callvirt instance class [mscorlib]System.Text.StringBuilder [mscorlib]System.Text.StringBuilder::Append(string)
IL_00be: pop
IL_00bf: ldloc.s dir
IL_00c1: ldarg.
IL_00c2: ldc.i4.
IL_00c3: add
IL_00c4: call class [mscorlib]System.Collections.Generic.List`<string> Client.Program::getAllFiles(string,
int32)
IL_00c9: stloc.s tmp
IL_00cb: ldloc.s tmp
IL_00cd: ldstr ""
IL_00d2: callvirt instance bool [mscorlib]System.Object::Equals(object)
IL_00d7: stloc.s CS$$
IL_00d9: ldloc.s CS$$
IL_00db: brtrue.s IL_00e8
IL_00dd: nop
IL_00de: ldloc.
IL_00df: ldloc.s tmp
IL_00e1: callvirt instance void class [mscorlib]System.Collections.Generic.List`<string>::AddRange(class [mscorlib]System.Collections.Generic.IEnumerable`<!>)
IL_00e6: nop
IL_00e7: nop
IL_00e8: nop
IL_00e9: ldloc.s CS$$
IL_00eb: ldc.i4.
IL_00ec: add
IL_00ed: stloc.s CS$$
IL_00ef: ldloc.s CS$$
IL_00f1: ldloc.s CS$$
IL_00f3: ldlen
IL_00f4: conv.i4
IL_00f5: clt
IL_00f7: stloc.s CS$$
IL_00f9: ldloc.s CS$$
IL_00fb: brtrue.s IL_008a
IL_00fd: nop
IL_00fe: ldloc.
IL_00ff: stloc.s CS$$
IL_0101: br.s IL_0103
IL_0103: ldloc.s CS$$
IL_0105: ret
} // end of method Program::getAllFiles

我加注释的地方,就是方法从头开始到第一个循环结束的地方,后面的循环和这个类似,就不解读了。

初读 c# IL中间语言的更多相关文章

  1. 读懂IL代码就这么简单(三)完结篇

    一 前言 写了两篇关于IL指令相关的文章,分别把值类型与引用类型在 堆与栈上的操作区别详细的写了一遍 这第三篇也是最后一篇,之所以到第三篇就结束了,是因为以我现在的层次,能理解到的都写完了,而且个人认 ...

  2. 读懂IL代码就这么简单 (一)

    一前言 感谢 @冰麟轻武 指出文章的错误之处,现已更正 对于IL代码没了解之前总感觉很神奇,初一看完全不知所云,只听高手们说,了解IL代码你能更加清楚的知道你的代码是如何运行相互调用的,此言一出不明觉 ...

  3. 读懂IL代码就这么简单

    原文地址:http://www.cnblogs.com/zery/p/3366175.html 一前言 感谢 @冰麟轻武 指出文章的错误之处,现已更正 对于IL代码没了解之前总感觉很神奇,初一看完全不 ...

  4. 【转载】读懂IL代码就这么简单(三)完结篇

    一 前言 写了两篇关于IL指令相关的文章,分别把值类型与引用类型在 堆与栈上的操作区别详细的写了一遍这第三篇也是最后一篇,之所以到第三篇就结束了,是因为以我现在的层次,能理解到的都写完了,而且个人认为 ...

  5. 【转载】读懂IL代码就这么简单 (一)

    一前言 感谢 @冰麟轻武 指出文章的错误之处,现已更正 对于IL代码没了解之前总感觉很神奇,初一看完全不知所云,只听高手们说,了解IL代码你能更加清楚的知道你的代码是如何运行相互调用的,此言一出不明觉 ...

  6. 读懂IL代码就这么简单(二)

    一 前言 IL系列 第一篇写完后 得到高人指点,及时更正了文章中的错误,也使得我写这篇文章时更加谨慎,自己在了解相关知识点时,也更为细致.个人觉得既然做为文章写出来,就一定要保证比较高的质量,和正确率 ...

  7. 读懂IL

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

  8. 轻松读懂IL

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

  9. 读懂IL代码就这么简单 ---- IL系列文章

    读懂IL代码就这么简单 (一) 读懂IL代码就这么简单(二) 读懂IL代码就这么简单(三)完结篇 出处:http://www.cnblogs.com/zery/tag/IL%20%E7%B3%BB%E ...

随机推荐

  1. C++ 函数模板“偏特化”

         模板是C++中很重要的一个特性,利用模板可以编写出类型无关的通用代码,极大的减少了代码量,提升工作效率.C++中包含类模板.函数模板,对于需要特殊处理的类型,可以通过特化的方式来实现特定类型 ...

  2. 三、scrapy后续

    CrawlSpiders 通过下面的命令可以快速创建 CrawlSpider模板 的代码: scrapy genspider -t crawl tencent tencent.com 我们通过正则表达 ...

  3. 验证SQLServer死锁进程

    SELECT '现在没有阻塞和死锁信息' AS message -- 循环开始WHILE @intCounter <= @intCountProperties BEGIN-- 取第一条记录 SE ...

  4. 端到端测试工具--testcafe

    写在前面 随着业务的增加,复杂性的增加,我们更需要保证页面不能出错,之前需要每次上线之前需要每次人工测试,如果有好多改动,为保证业务不出错,需要耗费更多的时间来测试,所以我们需要写一些测试来保证业务的 ...

  5. Linux(ubuntu)安装redis集群,redis集群搭建

    今天学习一下redis集群的搭建.redis在现在是很常用的数据库,在nosql数据库中也是非常好用的,接下来我们搭建一下redis的集群. 一.准备 首先我们要安装c语言的编译环境,我们要安装red ...

  6. 3.2 while 循环

    Python 编程中 while 语句用于循环执行程序,即在条件满足的情况下,循环执行某段代码.所以就需要在循环的代码块中设计一种使代码块循环执行一定次数后是while语句的条件不满足,从而中止whi ...

  7. mysql 学习心得1

    1由于不靠这玩意吃饭 估计不准备精读 顺便中文版也不用担心翻译问题 科科 大致翻了下=,= mysql的感觉怎么就是背命令.... 2DDL语句 定义数据 创建删除修改 create drop alt ...

  8. JVM笔记2-Java虚拟机内存管理简介

    java虚拟机内存管理图如下图所示: 1.线程共享区,是所有的线程所共用的,线程共享区有一下几个组成: 1.方法区: 1.运行时常量池,已经被虚拟机加载的类信息(1.类的版本信息,2.字段,3.方法, ...

  9. Effective Java 第三版——34. 使用枚举类型替代整型常量

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  10. 转:彻底搞清referrer和origin

    在http协议中有这两个字段,之前一直隐隐约约的觉得是,一种标记请求来源的方法(的确是),但是更细致的对这两个字段的比较却没有一个清楚的认识. referrer 到底是referer还是referre ...