一起谈.NET技术,关于ASP.NET与CLR相互关系的一些总结

简介:   CLR(COM服务器)  CLR作为一个COM服务器实现在MSCorWks.dll文件中。安装.NET Framework时,表示CLR的COM服务器被注册到Windows的注册表里。  MSCorEE.dll(垫片)  MSCorEE.dll的职责是判断创建何种版本的CLR。

  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 事件。

  2年多了,回过来再重新温习一次,感觉不错。

目录
相关文章
|
6月前
|
存储 开发框架 前端开发
asp.net与asp.net优缺点及示例
asp.net与asp.net优缺点及示例
|
1月前
|
开发框架 前端开发 .NET
进入ASP .net mvc的世界
进入ASP .net mvc的世界
29 0
|
1月前
|
开发框架 中间件 .NET
C# .NET面试系列七:ASP.NET Core
## 第一部分:ASP.NET Core #### 1. 如何在 controller 中注入 service? 在.NET中,在ASP.NET Core应用程序中的Controller中注入服务通常使用<u>依赖注入(Dependency Injection)</u>来实现。以下是一些步骤,说明如何在Controller中注入服务: 1、创建服务 首先,确保你已经在应用程序中注册了服务。这通常在Startup.cs文件的ConfigureServices方法中完成。例如: ```c# services.AddScoped<IMyService, MyService>(); //
63 0
|
1月前
|
开发框架 前端开发 .NET
C# .NET面试系列六:ASP.NET MVC
<h2>ASP.NET MVC #### 1. MVC 中的 TempData\ViewBag\ViewData 区别? 在ASP.NET MVC中,TempData、ViewBag 和 ViewData 都是用于在控制器和视图之间传递数据的机制,但它们有一些区别。 <b>TempData:</b> 1、生命周期 ```c# TempData 的生命周期是短暂的,数据只在当前请求和下一次请求之间有效。一旦数据被读取,它就会被标记为已读,下一次请求时就会被清除。 ``` 2、用途 ```c# 主要用于在两个动作之间传递数据,例如在一个动作中设置 TempData,然后在重定向到另
99 5
|
3月前
|
开发框架 安全 搜索推荐
分享105个NET源码ASP源码,总有一款适合您
分享105个NET源码ASP源码,总有一款适合您
27 4
|
7月前
|
开发框架 前端开发 .NET
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
176 0
.NET 8 Release Candidate 1 (RC1)现已发布,包括许多针对ASP.NET Core的重要改进!
|
8月前
|
存储 开发框架 .NET
ASP.NET学生管理系统(.NET毕业设计)
ASP.NET学生管理系统(.NET毕业设计)
100 0
|
9月前
|
开发框架 前端开发 JavaScript
ASP .Net Core 中间件的使用(一):搭建静态文件服务器/访问指定文件
ASP .Net Core 中间件的使用(一):搭建静态文件服务器/访问指定文件
|
10月前
|
开发框架 供应链 前端开发
net基于asp.net的社区团购网站
社区团购系统依托社区团购系统和社区门店,是现在的一个重大市场和发展方向,通过研究企业在社区团购系统环境下的营销模式创新,对于普通的零售业和传统社区团购系统的转型发展具有重要的理论意义。随着互联网行业的发展,人们的生活方式发生着重大变化,人们越来越倾向于网络购物,这对传统企业来说如何把客户留下是一个重大挑战。就现在而言,由于社区团购的竞争已经进入最紧张激烈的阶段,有些团购平台甚至已经彼此之间打起了价格战,其中不乏有平台因为利润变少或资金链断裂而半途败亡。企业在实际的商业活动中,往往会面临许多等待优化的问题。因此,要在竞争激烈的市场中拔得头筹,必须重视提升对新商业模式的全面认知,科学于实际贴合的分
|
缓存 前端开发 JavaScript
采用.Net Core技术框架开发的医院云LIS平台源码,B/S架构
基于B/S架构的医学实验室检验系统源码,整个系统的运行基于WEB层面,只需要在对应的工作台安装一个浏览器软件有外网即可访问。全套系统采用云部署模式,部署一套可支持多家医院检验科共同使用。 采用.Net Core新的技术框架、DEV报表、前端js封装、分布式文件存储、分布式缓存等,支持LIS独立部署,Docker部署等多种方式。