.Net Compact Framework CLR设计系列之JIT编译器讲解
创始人
2024-06-09 12:40:25
0

这个.Net Compact Framework CLR设计这个话题被很多人关注。现在我们可以从设计者的角度,深入了解.Net Compact Framework CLR设计的内部结构。这部分我们要讨论JIT编译器的知识。感谢Steven Pratschner,感谢他给我们带来了这么好的文章。

设计JIT编译器

.Net Compact Framework CLR设计的***章,以后我们还会为大家提供更多的,希望大家关注。.Net Compact Framework的JIT编译器与.Net Framework***的不同在于内存使用。在内存紧张的情况下,.Net Compact Framework可以释放Jitted代码,将内存返还给操作系统。正如你所预料的那样,如此设计的原因是因为用于存储jitted代码的堆是分配在应用程序私有的32MB地址空间上的(更多信息可以参考***部分)。除了私有地址空间非常小之外,考虑到它们从来不被分页,在内存受限设备上运行程序,必要时减少空间压力的设计是绝对必要的。

当程序被执行时,JIT编译器会在堆上分配内存,用来存储每个方法编译生成的本地代码。因为编译和内存分配发生在每个方法运行的时候,每次内存分配都会让堆相应减少。换句话说,就是JIT堆在小幅度地逐渐增长。在程序运行过程中,JIT堆会增长到很大的程度。在Compact Framework的早期版本中,JIT堆的尺寸被限制在一个固定的大小中。在第二版中,这个限制已经被去掉了,因此在新方法需要被编译时,堆会增加。

三种情况的发生,会导致JIT堆的大部分空间被释放并将内存归还给OS(这里只所以说“大部分空间”是因为Compact Framework必须始终保留当前执行应用程序方法的jitted代码)。首先,如果CLR试图分配更多内存时,收到一个来自操作系统的错误,JIT堆将会收缩。CLR会认为这个失败表明可用内存数量不足,于是尽可能多的挥手JIT堆中的代码。从JIT堆中释放本地代码的动作是根据代码存在期限决定的。其次,当一个程序被切换到后台时,代码会被回收。在Windows Mobile中,应用程序通常不会被关闭,但是会被切换到后台。当一个程序被切换到后台时,通过释放代码,CLR可以获得更多可用内存供前台程序使用,这样可以增加同时运行在设备上应用程序的数量。***,当一个托管应用程序收到来自Windows CE的WM_HIBERNATE消息时,CLR会回收jitted代码。当OS发现运行的系统资源过低时,会发出WM_HIBERNATE消息。当设备资源缺乏时,响应WM_HIBERNATE消息的代码回收是CLR释放内存和其他资源操作的一部分。

我在稍后章节讨论自动内存管理时,你将会看到,代码回收是整个垃圾收集的一部分。

代码回收

Figure 1The size of the JIT heap over the lifetime of an application.

图1中的一些情况是十分值得注意的。首先,图中的两个低点发生的时间,对应于程序被切换到后台和堆的尺寸太大而开始代码回收的时间。同样,注意程序启动的时候比程序从后台切换回来时,会有更多代码被jitted。这大概是因为应用程序包含一些初始化代码,而这些代码只是在程序开始时被调用。

因为CLR会在内存紧张或者程序切换到后台时丢弃本地代码,所以在程序继续运行时,相同的IL代码会被再次JIT编译。正因为如此我们才作出了第二个关于JIT编译器的设计决定:编译IL代码的时间通常优先于生成本地代码的质量。作为一个优秀的编译器,Compact Framework JIT编译器做了一些基本的优化,但是为了让应用程序保持响应,就需要更快地生成代码,更多地优化措施要根据其速度来决定是否执行。

JIT编译器***一个关键设计原则是不涉及内存使用,这样做是为了让JIT编译器更方便移植。我在***部分曾提到,Compact Framework的运行环境不仅要求它能够在内存受限设备上运行,而且需要它可以在不同的处理器上运行。.Net Compact Framework目前可以运行在包括x86、Arm、SH和MIPS等处理器上,而且还可以根据要求支持更多的处理器。因为需要跨越不同领域的设备,JIT编译器被设计成花费最少时间,便可以支持一种新的处理器类型的架构。一种技术被用来增加可移植性,就是将处理器相关的操作限制在最小程度。

Why no Native Images?
桌面版的.Net Framework使用一种叫“本地映像”的技术,当应用程序加载时,IL代码都需要被JIT编译器编译。如果利用了本地映像技术,应用程序可以更快地启动。本地映像是一个保存在硬盘上的文件,包含被编译好的IL代码。当.Net Framework安装时,它会调用JIT编译器生成mscorlib、System.Windows.Forms等类库的本地CPU指令。当应用程序启动时,就可以直接调用储存在本地映像中的已经生成的本地代码,从而节省了JIT编译这些程序集的时间。用户也可以为自己编写的程序集产生本地映像(可以参考.Net Framework SDK中ngen.exe工具的文档)。

 .Net Compact Framework没有使用本地映像的主要原因是他们的尺寸。根据本地指令集的不同,一个程序集被JIT编译后产生的本地代码大小大约是IL代码的三到四倍。在压缩之后,.Net Compact Framework类库大约是4.5MB。如果相应的本地映像是这个尺寸的四倍的话,你就会看到本地映像需要的存储空间是设备中可用内存中相当大的一部分。另外一种可能就是可以将本地映像保存在外部存储卡上。可是从存储卡读取本地映像文件的速度是非常慢的,因此我们不确定启动时间可以通过本地映像的方法可以缩短。

以上介绍.Net Compact Framework CLR设计,Compact Framework JIT团队会在每个主要版本中考虑本地映像,也许在CLR的未来某个版本中,我们会看到本地映像,或者类似的技术。

【编辑推荐】

  1. CLR Via C#教程之装箱和拆箱讲述
  2. CLR VIA C#教程之基元类型 值类型 引用类型介绍
  3. 趣谈CLR集成性能设计选择
  4. 实作CLR存储过程十四步
  5. 深入挖掘CLR内存管理机制原理

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...
《非诚勿扰》红人闫凤娇被曝厕所... 【51CTO.com 综合消息360安全专家提醒说,“闫凤娇”、“非诚勿扰”已经被黑客盯上成为了“木...
2012年第四季度互联网状况报... [[71653]]  北京时间4月25日消息,据国外媒体报道,全球知名的云平台公司Akamai Te...