这个系列文章的第一部分上了CSDN的首页,可见这个话题还是有很多人关注的。而且我们也能够从设计者的角度深入了解.Net Compact Framework CLR的内部结构。这部分我们要讨论JIT编译器的知识。再次感谢Steven Pratschner,感谢他给我们带来了这么好的文章,本章的英文版可以在这里找到。
第二章 设计JIT编译器
这是《设计.Net Compact Framework CLR》系列文章的第二部分。你可以在这里看到第一部分。在第二部分中,我们将讨论.Net Compact Framework JIT编译器的一些基本设计原则。
----
.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释放内存和其他资源操作的一部分。
我在稍后章节讨论自动内存管理时,你将会看到,代码回收是整个垃圾收集的一部分。
<shapetype id="_x0000_t75" stroked="f" filled="f" path="m@4@5l@4@11@9@11@9@5xe" o:preferrelative="t" o:spt="75" coordsize="21600,21600"><stroke joinstyle="miter"></stroke><formulas><f eqn="if lineDrawn pixelLineWidth 0"></f><f eqn="sum @0 1 0"></f><f eqn="sum 0 0 @1"></f><f eqn="prod @2 1 2"></f><f eqn="prod @3 21600 pixelWidth"></f><f eqn="prod @3 21600 pixelHeight"></f><f eqn="sum @0 0 1"></f><f eqn="prod @6 1 2"></f><f eqn="prod @7 21600 pixelWidth"></f><f eqn="sum @8 21600 0"></f><f eqn="prod @7 21600 pixelHeight"></f><f eqn="sum @10 21600 0"></f></formulas><path o:connecttype="rect" gradientshapeok="t" o:extrusionok="f"></path><lock aspectratio="t" v:ext="edit"></lock></shapetype><shape id="_x0000_i1025" style="WIDTH: 540.75pt; HEIGHT: 504.75pt" alt="" type="#_x0000_t75"><imagedata o:href="http://static.flickr.com/35/72984343_3298504d99_o.jpg" src="file:///C:/DOCUME~1/wolf/LOCALS~1/Temp/msoclip1/01/clip_image001.jpg"></imagedata></shape>
Figure 3
The size of the JIT heap over the lifetime of an application.
图3中的一些情况是十分值得注意的。首先,图中的两个低点发生的时间,对应于程序被切换到后台和堆的尺寸太大而开始代码回收的时间。同样,注意程序启动的时候比程序从后台切换回来时,会有更多代码被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。如果相应的本地映像是这个尺寸的四倍的话,你就会看到本地映像需要的存储空间是设备中可用内存中相当大的一部分。另外一种可能就是可以将本地映像保存在外部存储卡上。可是从存储卡读取本地映像文件的速度是非常慢的,因此我们不确定启动时间可以通过本地映像的方法可以缩短。
Compact Framework JIT团队会在每个主要版本中考虑本地映像,也许在CLR的未来某个版本中,我们会看到本地映像,或者类似的技术。
在下一章中,我们将看一下Compact Framework中的垃圾收集相关的设计原则。
This posting is provided "AS IS" with no warranties, and confers no rights.
分享到:
相关推荐
.NET Compact Framework 2.0是微软为嵌入式设备和移动平台开发的一个精简版.NET框架,主要用于掌上电脑(PDA)、智能手机和其他资源有限的设备。这个框架为开发者提供了在这些平台上构建和运行.NET应用程序的能力,...
.NET Compact Framework是微软为移动设备和嵌入式系统设计的一个精简版.NET Framework,它为开发者提供了一个在Windows Mobile、Windows CE(包括Wince)等平台上构建应用程序的强大环境。本书将深入探讨这一框架的...
此安装程序数据库中包含安装 Microsoft .NET Compact Framework 2.0 所需的逻辑和数据
WinCE和Windos Mobile(.Net Compact Framework)开发,包括《Windows CE.net内核定制及应用开发.pdf》、《Microsoft .NET mobiililaitteille: .NET Compact Framework.pdf》和《Microsoft .NET Compact Framework 3.5...
.net compact framework 自定义Radiobutton
.NET Compact Framework 3.5是微软为嵌入式设备和移动平台开发的应用程序提供的一套框架,它在.NET Framework的基础上进行了优化,以适应资源有限的环境。这个版本引入了许多新特性和改进,使得开发者能够在小型设备...
《.NET Compact Framework开发技巧大串烧》是MSDN举办的一场Webcast活动,主要针对的是使用.NET Compact Framework进行移动设备或嵌入式系统开发的程序员。在这场视频讲座中,专家深入浅出地讲解了一系列关于.NET ...
一个 .net compact framework 2.0 上的简单记事本(在Coolpad N900 的 WinCE6环境下可以运行) vb2005开发环境 包括 屏幕输入面板控制,剪贴板访问,文件读写等基础代码示例
5. **设备兼容性**:.NET Compact Framework设计时考虑了各种不同类型的设备,确保应用程序可以在各种硬件配置和操作系统版本上运行。 6. **安全性**:框架内置了安全性模型,以保护设备免受恶意软件攻击,同时也...
Delphi for .NET Compact Framework Technology PreviewThis technology preview supports both PocketPC and SmartPhone devices running .NET Compact Framework 1.x.Note: Some devices (such as SmartPhone) do ...
《.NET Compact Framework 3.5版的新特性》 .NET Compact Framework 3.5是微软为嵌入式设备和移动设备开发的一款精简版.NET框架,它为开发者提供了丰富的功能和改进,使得在有限资源的平台上开发高效、功能强大的...
《Windows Mobile .NET Compact Framework 用户通知示例详解》 Windows Mobile .NET Compact Framework是微软为移动设备开发的应用程序框架,它允许开发者使用.NET Framework的大部分功能来构建针对智能手机和平板...
WinCE开发工具 可用于WinCE6.0的开发 微软官网可以下载到
.NET Compact Framework是微软推出的一种专门用于嵌入式设备和移动平台的.NET框架,它为开发者提供了在小型设备上构建应用程序的能力。本串烧教程将深入探讨.NET Compact Framework的开发技巧,帮助开发者提升在有限...
.NET Compact Framework是微软专门为嵌入式设备和移动设备设计的一个.NET框架版本,它允许开发者在这些平台上构建应用程序,包括进行网络通信。本压缩包提供的源代码着重展示了如何在.NET Compact Framework下进行...
.NET Compact Framework是微软推出的一种专门针对嵌入式设备和移动计算平台的框架,它为开发者提供了在小型设备上构建应用程序的能力。在这个".NET Compact Framework开发技巧大串烧"的资源中,我们很可能会深入探讨...
首先,.NET Compact Framework是微软为智能设备设计的一个精简版.NET Framework,它在桌面和笔记本电脑上广泛使用的.NET Framework的基础上进行了优化,以适应资源有限的移动设备。这个框架允许开发者使用C#、VB.NET...