原文地址:http://www.cnblogs.com/jasenkin/archive/2010/10/20/asp-net-clr-relation.html

CLR(COM服务器)

CLR作为一个COM服务器实现在MSCorWks.dll文件中。安装.NET Framework时,表示CLR的COM服务器被注册到Windows的注册表里。

MSCorEE.dll(垫片)

MSCorEE.dll的职责是判断创建何种版本的CLR。 非托管应用程序宿主调用MSCorEE.dll(shim)中CorBindToRuntimeEx函数或者另一个相似的函数来创建CLR COM的实例。

一台机器可以安装多个版本的CLR,但在机器中只有一个版本的MSCorEE.dll文件。

以上两者之间的关系如下代码所示(c++):     MSCorEE.dll(垫片)---->CLR(COM服务器)


#include <Windows.h>
#include <MSCorEE.h>
#include <stdio.h>
#pragma comment(lib,"mscoree.lib")
int main(int argc, CHAR* argv[])
{
    ICLRRuntimeHost *pClrHost;
    //调用<MSCorEE.h>中的CorBindToRuntimeEx()生成COM服务器
    HRESULT hr= CorBindToRuntimeEx(NULL, NULL, NULL, CLSID_CLRRuntimeHost, IID_ICLRRuntimeHost, (LPVOID*)&pClrHost); 
    //启动clr
    pClrHost->Start();   
     //............................其他操作   
     //终止clr
     pClrHost->Stop();
     getchar();
     return 0;
}

应用程序域(AppDomain)

一个应用程序域是一组程序集的一个容器。应用程序域的全部目的就是提供隔离性。

上图为一个单独的Windows进程,该进程中运行着一个CLR COM服务器。该CLR中每个应用程序域都有自己的加载器堆。(系统域,共享域,默认域#1,其他域#2)

一些特殊的程序集如MSCorLib.dll,它需要被所有的应用程序域共享,那么他就以一种对应用程序域保持中立的方式被加载,由CLR维护一个特殊的加载器堆。

线程与应用程序域的关系

线程与应用程序域没有一对一的关系。多个应用程序域可以位于一个Windows进程中,所以一个线程可以执行一个应用程序域中的代码,然后又执行另一个应用程序域中的代码。从CLR的角度看,线程每次只能执行一个应用程序域中的代码。线程可以通过Thead的静态方法GetDomain来请求CLR当前正在那个应用程序域中执行。

关于卸载的问题

一旦CLR加载到Windows的进程,就永远不会被卸载,要想卸载进程中的CLR,只能终止进程,导致Windows清理该进程使用的所有资源。

共享域的程序集永远不会被卸载,回收它们的唯一方式就是终止Windows进程。

卸载一个应用程序域(默认域#1,其他域#2)可以导致CLR卸载应用程序域中的所有程序集,并且释放应用程序域的加载器堆。

ASP.NET 应用程序

  ASP.NET是一个ISAPI DLL(实现于ASPNET_ISAPI.DLL中)。

 ①当客户端请求一个由ASP.NET ISAPI DLL处理的URL时(第一次),ASP.NET会创建一个工作进程(ASPNET_wp.exe / w3wp.exe即宿主,工作进程是一个寄宿有CLR COM服务器的Windows进程)。

CLR为该Web程序创建一个新的应用程序域。(参考上面的图)

CLR将必要的程序集加载到新建的应用程序域中。

CLR创建一个Web应用程序类型的实例,调用其中方法响应客户端请求。随着Web应用程序的运行,如果代码中引用到了更多类型,CLR会继续将必要的程序集加载到当前Web应用程序的应用程序域中。

  当更多的客户端向一个已经运行的Web应用程序发出请求时,ASP.NET不会再去创建新的应用程序域,相反,它会使用现有的应用程序域,创建一个新的Web应用程序类型的实例,并调用其中的方法。这些方法被JIT编译成了本地代码,所以后续客户端的请求处理性能将会有所提高。

看下我们的工作进程,如下任务管理器截图中所示:

 

ASP.NET 应用程序生命周期的各个阶段
(一)用户从 Web 服务器请求应用程序资源(映射文件扩展名)

ASP.NET 应用程序的生命周期以浏览器向 Web 服务器发送请求为起点。ASP.NET 是 Web 服务器下的 ISAPI 扩展。Web 服务器接收到请求时,会对所请求的文件的文件扩展名进行检查,确定应由哪个 ISAPI 扩展处理该请求,然后将该请求传递给合适的 ISAPI 扩展。ASP.NET 处理已映射到其上的文件扩展名,如 .aspx、.ascx、.ashx 和 .asmx。

(二)ASP.NET 接收对应用程序的第一个请求。
     当 ASP.NET 接收到对应用程序中任何资源的第一个请求时,名为 ApplicationManager 的类会(1)创建一个应用程序域。在应用程序域中,(2)将HostingEnvironment 类创建一个实例,该实例提供对有关应用程序的信息(如存储该应用程序的文件夹的名称)的访问。下面的关系图说明了这种关系:

  运行ASP.NET的进程为W3WP.EXE

编写下述代码,我们可以观察一下HostingEnvironment 的各种属性:


 1 <script runat="server">
 2   protected void Page_Load(object sender, EventArgs e)
 3   {
 4       StringBuilder sb = new StringBuilder();
 5       sb.Append(System.Web.Hosting.HostingEnvironment.ApplicationID).Append("<br/>")
 6           .Append(System.Web.Hosting.HostingEnvironment.ApplicationPhysicalPath).Append("<br/>")
 7            .Append(System.Web.Hosting.HostingEnvironment.ApplicationVirtualPath).Append("<br/>")
 8             .Append(System.Web.Hosting.HostingEnvironment.SiteName).Append("<br/>")
 9              .Append(System.Web.Hosting.HostingEnvironment.ApplicationHost.ToString());      
10       Response.Write(sb.ToString());    
11   }
12 </script>

输出内容为:

a215d83d                                                   //ApplicationID
E:\MVC\Jasen\Jasen.Web\                                    //存储该应用程序的文件夹的名称
/
默认网站
System.Web.Hosting.SimpleApplicationHost 

(三)为每个请求创建 ASP.NET 核心对象。
     ASP.NET 创建并初始化核心对象,如 HttpContext、HttpRequest 和 HttpResponse。HttpContext 类包含特定于当前应用程序请求的对象,如 HttpRequest 和 HttpResponse 对象。HttpRequest 对象包含有关当前请求的信息,包括 Cookie 和浏览器信息。HttpResponse 对象包含发送到客户端的响应,包括所有呈现的输出和 Cookie。
 
(四)将 HttpApplication 对象分配给请求
     初始化所有核心应用程序对象之后,将通过创建 HttpApplication 类的实例启动应用程序。如果应用程序具有 Global.asax 文件,则 ASP.NET 会创建 Global.asax 类的一个实例,并使用该派生类表示应用程序。MVC  Global.asax文件内容如下:


 1  public class MvcApplication : System.Web.HttpApplication
 2     {
 3         protected void Application_Start()
 4         {
            //AreaRegistration.RegisterAllAreas();
            //RegisterGlobalFilters(GlobalFilters.Filters);
            //ViewEngines.Engines.Add(new ViewEngine());
            //RegisterRoutes(RouteTable.Routes);
 5             //请求 ASP.NET 应用程序中第一个资源时调用。在应用程序的生命周期期间仅调用一次 
 6             //Application_Start 方法。可以使用此方法执行启动任务,如将数据加载到缓存中以及初始化静态值。 
 7 
 8             //在应用程序启动期间应仅设置静态数据。
 9         }
10 
11         public override void Init()
12         {
13             base.Init();//在创建了所有模块之后,对 HttpApplication 类的每个实例都调用一次。
14         }
15 
16         public override void Dispose()
17         {
18             base.Dispose();//在销毁应用程序实例之前调用。可使用此方法手动释放任何非托管资源。
19         }
20 
21         public void Application_End()
22         {
23             //在卸载应用程序之前对每个应用程序生命周期调用一次。
24         }
25     }

HttpApplication 进程的一个实例每次只处理一个请求。Application_Start 和 Application_End 方法,在应用程序域的生命周期期间,ASP.NET 仅调用这些方法一次,而不是对每个 HttpApplication 实例都调用一次。

(五)由 HttpApplication 管线处理请求。在处理该请求时将由 HttpApplication 类执行以下事件。

1.对请求进行验证,将检查浏览器发送的信息,并确定其是否包含潜在恶意标记。

2.如果已在 Web.config 文件的 UrlMappingsSection 节中配置了任何 URL,则执行 URL 映射。

3.引发 BeginRequest 事件。

4.引发 AuthenticateRequest 事件。引发 PostAuthenticateRequest 事件。

5.引发 AuthorizeRequest 事件。引发 PostAuthorizeRequest 事件。

6.引发 ResolveRequestCache 事件。引发 PostResolveRequestCache 事件。

7.根据所请求资源的文件扩展名(在应用程序的配置文件中映射),选择实现 IHttpHandler 的类,对请求进行处理。如果该请求针对从 Page 类派生的对象(页),并且需要对该页进行编译,则 ASP.NET 会在创建该页的实例之前对其进行编译。

8.引发 PostMapRequestHandler 事件。

-----------------------------------------------------------------------------------------------------------------------

9.引发 AcquireRequestState 事件。引发 PostAcquireRequestState 事件。

10.引发 PreRequestHandlerExecute 事件。

11.为该请求调用合适的 IHttpHandler 类的 ProcessRequest 方法。例如,如果该请求针对某页,则当前的页实例将处理该请求。             一般处理文件ashx的ProcessRequest():速度较快

12.引发 PostRequestHandlerExecute 事件。

13.引发 ReleaseRequestState 事件。引发 PostReleaseRequestState 事件。

-----------------------------------------------------------------------------------------------------------------------

14.如果定义了 Filter 属性,则执行响应筛选。

15.引发 UpdateRequestCache 事件。引发 PostUpdateRequestCache 事件。

16.引发 EndRequest 事件。

17.引发 PreSendRequestHeaders 事件。

18.引发 PreSendRequestContent 事件。

关于ASP.NET与CLR相互关系的一些总结的更多相关文章

  1. .Net 类型、对象、线程栈、托管堆运行时的相互关系

    JIT(just in time)编译器 接下来的会讲到方法的调用,这里先讲下JIT编译器.以CLR书中的代码为例(手打...).以Main方法为例: static void Main(){ Cons ...

  2. 【.Net基础一】 类型、对象、线程栈、托管堆运行时的相互关系

    目前在看CLR via C#,把总结的记下来,索性就把他写成一个系列吧. 1.[.Net基础一] 类型.对象.线程栈.托管堆运行时的相互关系 2.[.Net基础二]浅谈引用类型.值类型和装箱.拆箱 J ...

  3. 【转载】C#之C#、.NET Framework、CLR的关系

    C#..NET Framework.CLR的关系 很多人没有将C#..NET Framework(.NET框架).CLR(Common Language Runtime,公共语言运行库)这三者之间的关 ...

  4. H.264中NAL、Slice与frame意思及相互关系

    H.264中NAL.Slice与frame意思及相互关系 NAL nal_unit_type中的1(非IDR图像的编码条带).2(编码条带数据分割块A).3(编码条带数据分割块B).4(编码条带数据分 ...

  5. 斯坦福大学公开课机器学习: advice for applying machine learning | regularization and bais/variance(机器学习中方差和偏差如何相互影响、以及和算法的正则化之间的相互关系)

    算法正则化可以有效地防止过拟合, 但正则化跟算法的偏差和方差又有什么关系呢?下面主要讨论一下方差和偏差两者之间是如何相互影响的.以及和算法的正则化之间的相互关系 假如我们要对高阶的多项式进行拟合,为了 ...

  6. Anaconda、Miniconda、Conda、pip的相互关系_我是刘振岗_新浪博客

    Anaconda.Miniconda.Conda.pip的相互关系_我是刘振岗_新浪博客 http://blog.sina.com.cn/s/blog_8a122dcf0102x9vn.html

  7. (oralce)pga_aggregate_target与workarea_size_policy相互关系验证

    pga_aggregate_target与workarea_size_policy相互关系验证 先放上结论: 1. 当pga_aggregate_target设置为非0, 手工设置(workarea ...

  8. hibernate中的java对象有几种状态,其相互关系如何(区别和相互转换)。

    hibernate中的java对象有几种状态,其相互关系如何(区别和相互转换). 解答:在Hibernate中,对象有三种状态:临时状态.持久状态和游离状态. 临时状态:当new一个实体对象后,这个对 ...

  9. C#、.NET Framework、CLR的关系

    很多人没有将C#..NET Framework(.NET框架).CLR(Common Language Runtime,公共语言运行库)这三者之间的关系区分清楚,认为其版本号是一一对应的.其实不然,. ...

随机推荐

  1. 个人对js闭包的理解

      闭包算是前端面试的基础题,但我看了很多关于闭包的文章博客,但感觉很多对于闭包的理想还是有分歧的,现在网上对闭包的理解一般是两种: 有些文章认为闭包必须要返回嵌套函数中里面用到外面函数局部变量的方法 ...

  2. SqlSever基础 top 后面的n 由一个简单的计算式子计算得出

    镇场诗:---大梦谁觉,水月中建博客.百千磨难,才知世事无常.---今持佛语,技术无量愿学.愿尽所学,铸一良心博客.------------------------------------------ ...

  3. WINCE+6410 拨号上网

    我们现在的模块用的是USB接口,然后模拟成一个COM6进行通信,在CE的: 控制面板->网络连接 里要新建一个连接,选择"拨号连接"然后设置好区域代码,还有附加解调器命令+C ...

  4. JS cookie的使用

    js设置cookie有很多种方法. 第一种:(这个是w3c官网的代码) <script> //设置cookie function setCookie(cname, cvalue, exda ...

  5. jquery之event与originalEvent的关系、event事件对象用法浅析

    在jquery中,最终传入事件处理程序的 event 其实已经被 jQuery 做过标准化处理, 其原有的事件对象则被保存于 event 对象的 originalEvent 属性之中, 每个 even ...

  6. linux 查找替换

    esc:   命令模式与插入模式的切换 一.vi查找:    当你用vi打开一个文件后,因为文件太长,如何才能找到你所要查找的关键字呢?在vi里可没有菜单-〉查找, 不过没关系,你在命令模式下敲斜杆( ...

  7. 验证SMB登入

    使用SMB登入扫描器对大量主机的用户名和口令进行猜解,不过扫描动静很大,容易被察觉,而且每一次登入尝试都会被扫描的主机系统日志记录下来,留下痕迹不建议使用. 实例 第一步: msf > use ...

  8. Android——学习:线性布局权重分配

    LinearLayout在Android中被广泛使用,LinearLayout有一个比较重要的属性——android:layout_weight.按照字面理解就是该控件的权重,这个值默认是 零(0). ...

  9. hdu 5643 King's Game 约瑟夫变形

    首先约瑟夫问题的数学推理过程:我们知道第一个人(编号一定是(m-1) mod n) 出列之后,剩下的n-1个人组成了一个新的约瑟夫环(以编号为k=m mod n的人开始):k k+1 k+2 ... ...

  10. 装了maven插件的eclipse中M2_REPO无法编辑、删除(转)

    今天用了新版本的eclipse,用maven在命令行生成了一个普通项目.导入eclipse之后发现本地仓库的路径不正确. 显示的为 user.path/.m2/repository 但是我的仓库早已经 ...