用Visual Studio等IDE写C#的Hello World非常简单,但脱离了IDE你能不能打印出Hello World呢?这不是说工作时脱离IDE,而是学习一下CLR的执行模型.

  Hello World

  1. 新建一个记事本,输入如下代码,另存为HelloWorld.txt。
    using System;
    namespace HelloWorld
    {
       class Program
       {
            static void Main(string[] args) {
                Console.WriteLine("Hello World!");
                Console.ReadKey();
            }
        }
    }
  2. 打开Visual Studio 2008(2005,2010) 命令提示程序 

    image
  3. 切换到HelloWorld.txt的目录 
    image
  4. 运行命令:csc /out:Hello.exe HelloWorld.txt 
    image

  如无意外,将会编译出Hello.exe,能打印出Hello World。

  CLR执行模型-编译期

  CLR程序的执行过程大致分为两步,编译期和运行期,编译期过程大致如下图:

image

  其中编译期逻辑上也可分为两步:

  1. CLR(C#)编译器接受源代码文件,并编译为托管模块。托管模块包括IL代码、元数据、CLR头等组成部分。上面的例子中就是将HelloWorld.txt编译成托管模块。
  2. 一般程序集都会包含很多源代码文件(这里只有HelloWorld.txt)和资源文件,第二步就是把各个源代码文件和资源文件对应编译结果合并成程序集。

  执行上面两步就可以得到一个XX.dll或XX.exe的程序集,就像上面的Hello.exe。

  编译器如何知道要编译成托管模块还是资源文件?其实是必须明确告诉编译器每个文件的怎么编译,这个对应Visual Studio的文件属性的生成操作.

  右击任何Visual Studio解决资源方案的文件-->属性-->生成操作:

image

  指定Class1为嵌入的资源,用ILSpy查看会发现只是把Class1嵌入到程序集中,名称为:命名空间.文件名:

image

  你甚至可以将一张图片设为编译让编译器试图去编译它,不过会报错。

  运行期

  上面生成了程序集,程序集内的是IL代码,它还不是可运行的代码。IL是与CPU无关的机器语言,直到程序集被调用,才会由JIT(Just-in-Time,实时)编译器编译为本机代码(CPU指令)。在运行时,CLR执行如下步骤:

  1. 检查程序集的安全特性;
  2. 在内存中分配空间;
  3. 把程序集中的可执行代码发送给JIT编译器,把其中一部分编译成本机代码(CPU指令)。

  程序集的可执行代码在需要的时候由JIT编译器编译,然后本机代码(CPU指令)就被缓存以备后来的程序中执行。一旦应用程序终止,编译好的本机代码也会被丢弃。

  例如如果将上面的代码改为:

static void Main(string[] args) {
    Console.WriteLine("Hello");
    Console.WriteLine("World!");
    Console.ReadKey();
}

  第一个WriteLine需要先JIT编译,再执行。而由于已编译WriteLine的代码,所以第二个WriteLine会直接执行内存块中的代码,跳过JIT编译。

  由于分配内存、JIT编译过程等,所以程序会在第一次运行时造成一些性能损失,写ASP.NET时这种感觉特变明显,按了F5会等很久才会显示首页。

  下面模拟感受这个过程。用一大堆类延长内存分配的时间,参考这个文件HelloWorld.cs(博客园不支持txt格式):

image

  再次运行命令:csc /out:Hello.exe HelloWorld.txt,得到Hello.exe,执行时发现有一定的延迟才会打印出Hello World。

  生成本机代码

  使用.NET提供的NGen.exe,可以将IL代码编译成本机代码,可以解决上面的问题。NGen.exe有两个作用:

  1. 加快应用程序的启动速度。因为代码已编译为本机代码,运行时不需要再花时间编译。
  2. 减少应用程序的程序集。如果一个程序集会同时加载多个进程,NGen.exe会将IL编译成本机代码,并保存到一个单独的文件中。这样就可以通过"内存映射"的方式,同时映射到多个进程中,使代码共享,避免每个进程一份代码。

  再次运行 Visual Studio 2008(2005,2010) 命令提示程序

  运行如下命令:ngen install Hello.exe:

image
  命令完成(在我的机器大概要10秒左右,到能再次输入命令才完成)后,运行Hello.exe会发现马上就能打印出Hello World,没有任何延迟。

posted on 2015-05-21 08:48  GC2013  阅读(647)  评论(1编辑  收藏  举报