MSIL入门(一)C#代码与IL代码对比
基础概念
Microsoft中间语言(MSIL),也成为通用中间语言(CIL),是一组与平台无关的指令,由特定于语言的编译器从源代码生成。MSIL是独立于平台的,因此,他可以在任何公共语言基础架构支持特定的环境上执行。
通过JIT编译器将MSIL转换为特定计算机环境的特定机器代码。这是在执行MSIL之前完成的,同样,MSIL在需求的基础上转换为机器代码,既JIT编译器根据需要而不是整个MSIL进行编译。
公共语言运行时(CLR)中的执行过程:执行过程包括创建MSIL以及通过JIT编译器将MSIL转换为机器代码,如下所示:
在CLR的编译期间,特定于语言的编译器会将源代码转换为MSIL。此外,与MSIL一起,在编译中还会生成元数据。元数据包含诸如代码中类型的定义和签名,运行时信息等信息。
通过组装MSIL,可以创建一个公共语言基础结构(CLI)组装。该程序集基本上是用于安全性,部署,版本控制等已编译的代码库,它具有两种类型,进程程序集(EXE)和程序集(DLL)。
然后JIT编译器将Microsoft中间语言(MSIL)转换为特定于JIT编译器运行所在的计算机环境的机器代码。MSIL在需求的基础上转换器为机器代码,即JIT编译器根据需要而不是整个MSIL进行编译。
然后,由JIT编译器获得的机器代码由计算机的处理器执行。
打印字符串
static void Main(string[] args){ Console.WriteLine("HelloWord!");}
.entrypoint 定义程序的入口点,该函数在程序启动时由.NET运行库调用
.maxstack 定义函数代码所用堆栈的最大深度
.ldstr string把一个字符串常量装入堆栈
call function(parameters)调用静态函数,函数的参数必须在函数调用前装入堆栈
pop 取出栈顶的值,当我们不需要把值存入变量时使用
ret 从当前方法返回,并将返回值(如果存在)从调用方的计算堆栈推送到被调用方的计算堆栈上。
.method private hidebysig static void Main( string[] args ) cil managed { .entrypoint //主函数,程序的入口 .maxstack 8 //栈的最大深度。 // [8 9 - 8 10] IL_0000: nop //什么都不做 // [9 13 - 9 45] IL_0001: ldstr "HelloWord!" //把字符串压入堆栈 IL_0006: call void [System.Console]System.Console::WriteLine(string) //调用WriteLine IL_000b: nop //什么都不做 // [10 9 - 10 10] IL_000c: ret //return } // end of method Program::Main
数据运算
add 2个值相加。命令的参数必须在调用前装入堆栈,该函数从堆栈中移除参数并把运算后的结果压入堆栈。
sub 2个值相减
mul 2个值相乘
static void Main(string[] args){ int x = 1; Console.WriteLine(x * 3 + 1 - 1);}
.method private hidebysig static void Main( string[] args ) cil managed { .entrypoint //主函数,程序的入口 .maxstack 2 //栈的最大深度 .locals init ( [0] int32 x, [1] int32 y, [2] int32 z //本地变量定义,定义int类型的 x、y、z ) // [8 9 - 8 10] IL_0000: nop //什么都不做 // [9 13 - 9 23] IL_0001: ldc.i4.1 //把x的值放到计算堆栈上 IL_0002: stloc.0 //把计算堆栈顶部的值(x)放到调用堆栈索引0处 // [10 13 - 10 23] IL_0003: ldc.i4.3 //把z的值放到计算堆栈上 IL_0004: stloc.1 //把计算堆栈顶部的值(y)放到调用堆栈索引1处 // [11 13 - 11 23] IL_0005: ldc.i4.1 //把x的值放到计算堆栈上 IL_0006: stloc.2 //把计算堆栈顶部的值(z)放到调用堆栈索引2处 // [12 13 - 12 46] IL_0007: ldloc.0 //把调用堆栈索引为0处的值复制到计算堆栈 IL_0008: ldloc.1 //把调用堆栈索引为1处的值复制到计算堆栈 IL_0009: mul //相乘 IL_000a: ldloc.2 //把调用堆栈索引为2处的值复制到计算堆栈 IL_000b: add //相加 IL_000c: ldloc.2 //把调用堆栈索引为2处的值复制到计算堆栈 IL_000d: sub //相减 IL_000e: call void [System.Console]System.Console::WriteLine(int32) //调用WriteLine IL_0013: nop //什么都不做 // [13 9 - 13 10] IL_0014: ret //return } // end of method Program::Main