MVC之前-ASP.NET初始化流程分析1
理论的解说一般都是枯燥的,研究一个代码框架更是如此,似乎除了对大量的源码加以解释之外无话可说,但是这又是不可缺少的,否则应用这样的框架起来总有不放心之感,总有不少的坑要踩。更进一步讲,一个好的框架应该给它的使用者足够的扩展空间(尤其是像MVC这样的基础框架),对框架本身没有很好的把握则对于框架的扩展似乎就只能寻找别人的“成功”经验了,然而生搬硬套从来不是解决问题的好方法,合理运用他人经验与创造性的解决方案都离不开对框架的深入理解,从这方面来说代码分析其实强于抽象的说明,因为代码的逻辑是确定的,代码的语言说服力其实强于书面语言(当然是指编写良好的代码)。当然对于代码之外的讨论也是非常必要甚至更加重要的,比如探讨框架设计的思路和模式,框架应用的相关实践乃至对框架的扩展等等。此一系列文章都是基于Asp.net Mvc框架的源码(包括Asp.net的一部分)对框架的基本设计与实现作出分析,另外参考了自己的一些项目以及当前一些优秀的开源项目(比如orchard)来讨论框架的应用与扩展。
Asp.net Mvc是当前使用比较多的web框架,也是比较先进的框架,目前.net的大部分源码都已经开放,这大大方便了我们对Asp.net Mvc的分析,下面就从Http请求进入Mvc框架处理之前的基本流程说起。
由于各IIS版本和工作模式(经典模式、集成模式)的不同,Http请求进入Asp.net的处理通道并不一样,这里不去细究里面的细节,就从创建应用程序域开始:
AppManagerAppDomainFactory分析
注:AppDomainFactory及AppManagerAppDomainFactory类在System.Web.Hosting中实现
在创建Appdomain时会调用IAppDomainFactory接口,该接口的实现如下:
public sealed class AppDomainFactory : IAppDomainFactory {
private AppManagerAppDomainFactory _realFactory;
public AppDomainFactory() {
_realFactory = new AppManagerAppDomainFactory();
}
public Object Create(String module, String typeName, String appId, String appPath,String strUrlOfAppOrigin, int iZone) {
return _realFactory.Create(appId, appPath);
}
}
该实现会调用AppManagerAppDomainFactory完成实际的创建过程。
public sealed class AppManagerAppDomainFactory : IAppManagerAppDomainFactory {
private ApplicationManager _appManager;
public AppManagerAppDomainFactory() {
_appManager = ApplicationManager.GetApplicationManager();
_appManager.Open();
}
public Object Create(String appId, String appPath) {
try {
if (appPath[0] == '.') {
System.IO.FileInfo file = new System.IO.FileInfo(appPath);
appPath = file.FullName;
}
if (!StringUtil.StringEndsWith(appPath, '\\')) {
appPath = appPath + "\\";
}
ISAPIRuntime isapiRuntime = (ISAPIRuntime)_appManager.CreateObjectInternal(appId, typeof(ISAPIRuntime), appHost,false, null);
isapiRuntime.StartProcessing();
return new ObjectHandle(isapiRuntime);
}
catch (Exception e) {
Debug.Trace("internal", "AppDomainFactory::Create failed with " + e.GetType().FullName + ": " + e.Message + "\r\n" + e.StackTrace);
throw;
}
}
}
代码的主要作用,就是通过ApplicationManager的CreateObjectInternal创建AppDomain,创建HostingEnvironment等,最终获取ISAPIRuntime的实例,然后让非托管代码调用。
ApplicationManager分析
注:ApplicationManager类在System.Web.Hosting中实现
首先看ApplicationManager类中的静态的创建ApplicationManager对象的方法:
public static ApplicationManager GetApplicationManager() {
if (_theAppManager == null) {
lock (_applicationManagerStaticLock) {
if (_theAppManager == null) {
if (HostingEnvironment.IsHosted)
_theAppManager = HostingEnvironment.GetApplicationManager();
if (_theAppManager == null)
_theAppManager = new ApplicationManager();
}
}
}
return _theAppManager;
}
如果HostingEnvironment已经创建则可以直接返回当前HostingEnvironment设置的ApplicationManager,如果没有则创建新的ApplicationManager实例。
然后看ApplicationManager的CreateObjectInternal方法:
internal IRegisteredObject CreateObjectInternal(String appId, Type type, IApplicationHost appHost, bool failIfExists, HostingEnvironmentParameters hostingParameters) {
if (!typeof(IRegisteredObject).IsAssignableFrom(type))
throw new ArgumentException(SR.GetString(SR.Not_IRegisteredObject, type.FullName), "type");
HostingEnvironment env = GetAppDomainWithHostingEnvironment(appId, appHost, hostingParameters);
ObjectHandle h = env.CreateWellKnownObjectInstance(type.AssemblyQualifiedName, failIfExists);
return (h != null) ? h.Unwrap() as IRegisteredObject : null;
}
首先要先取得HostingEnvironment的实例,然后通过该实例的CreateWellKnownObjectInstance方法返回上述Create方法需要的ISAPIRuntime的实例。先来看如何获取HostingEnvironment实例的方法GetAppDomainWithHostingEnvironment。
private HostingEnvironment GetAppDomainWithHostingEnvironment(String appId, IApplicationHost appHost, HostingEnvironmentParameters hostingParameters) {
LockableAppDomainContext ac = GetLockableAppDomainContext (appId);
lock (ac) {
HostingEnvironment env = ac.HostEnv;
if (env != null) {
try {
env.IsUnloaded();
}
catch(AppDomainUnloadedException) {
env = null;
}
}
if (env == null) {
env = CreateAppDomainWithHostingEnvironmentAndReportErrors(appId, appHost, hostingParameters);
ac.HostEnv = env;
Interlocked.Increment(ref _accessibleHostingEnvCount);
}
return env;
}
}
首先会检查字典是否会有已经存在的HostingEnvironment实例,如果有就返回,没有就会创建一个新的并保存到字典中,查看相关代码发现最终会调用ApplicationManager的私有方法CreateAppDomainWithHostingEnvironment创建AppDomain和HostingEnvironment。
private HostingEnvironment CreateAppDomainWithHostingEnvironment(String appId, IApplicationHost appHost, HostingEnvironmentParameters hostingParameters)
{
……
appDomain = AppDomain.CreateDomain(domainId,GetDefaultDomainIdentity(),setup);
……
Type hostType = typeof(HostingEnvironment);
String module = hostType.Module.Assembly.FullName;
String typeName = hostType.FullName;
ObjectHandle h = null;
……
try {
h = Activator.CreateInstance(appDomain, module, typeName);
}
……
HostingEnvironment env = (h != null) ? h.Unwrap() as HostingEnvironment : null;
if (env == null)
throw new SystemException(SR.GetString(SR.Cannot_create_HostEnv));
IConfigMapPathFactory configMapPathFactory = appHost.GetConfigMapPathFactory();
if (appDomainStartupConfigurationException == null) {
env.Initialize(this, appHost, configMapPathFactory, hostingParameters, policyLevel);
}
else {
env.Initialize(this, appHost, configMapPathFactory, hostingParameters, policyLevel, appDomainStartupConfigurationException);
}
return env;
}
可以看到代码创建了AppDomain和HostingEnvironment实例,创建HostingEnvironment实例以后,紧接着会调用其Initialize方法来进行初始化,然后返回对象实例。
HostingEnvironment分析
Initialize初始化方法,注意该方法的第一个参数是this,也就是ApplicationManager实例自身,这样会在HostingEnvironment中设置ApplicationManager,因而初始化之后可以通过HostingEnvironment获取ApplicationManager 。Initialize方法调用HttpRuntime的静态方法,进行一些初始化工作(其中会调用BuildManager的InitializeBuildManager方法进行初始化另外一些工作,其中包括编译App_Code目录下所有的.NET源代码)。最后如果HostingEnvironment初始化失败时会设置hostingInitFailed为true。
internal void Initialize(ApplicationManager appManager, IApplicationHost appHost, IConfigMapPathFactory configMapPathFactory, HostingEnvironmentParameters hostingParameters, PolicyLevel policyLevel, Exception appDomainCreationException)
{
……
_appManager = appManager;
……
HttpRuntime.InitializeHostingFeatures(hostingFlags,policyLevel,appDomainCreationException);
……
catch (Exception e) {
_hostingInitFailed = true;
}
}
创建HostingEnvironment后,会调用其方法CreateWellKnownObjectInstance创建ISAPIRuntime。创建好ISAPIRuntime实例后初始化工作就完成了第一个阶段,整个过程如下图所示:
理论的解说一般都是枯燥的,研究一个代码框架更是如此,似乎除了对大量的源码加以解释之外无话可说,但是这又是不可缺少的,否则应用这样的框架起来总有不放心之感,总有不少的坑要踩。更进一步讲,一个好的框架应该给它的使用者足够的扩展空间(尤其是像MVC这样的基础框架),对框架本身没有很好的把握则对于框架的扩展似乎就只能寻找别人的“成功”经验了,然而生搬硬套从来不是解决问题的好方法,合理运用他人经验与创造性的解决方案都离不开对框架的深入理解,从这方面来说代码分析其实强于抽象的说明,因为代码的逻辑是确定的,代码的语言说服力其实强于书面语言(当然是指编写良好的代码)。当然对于代码之外的讨论也是非常必要甚至更加重要的,比如探讨框架设计的思路和模式,框架应用的相关实践乃至对框架的扩展等等。此一系列文章都是基于Asp.net Mvc框架的源码(包括Asp.net的一部分)对框架的基本设计与实现作出分析,另外参考了自己的一些项目以及当前一些优秀的开源项目(比如orchard)来讨论框架的应用与扩展。
MVC之前-ASP.NET初始化流程分析1的更多相关文章
- ASP.NET初始化流程分析2
上一篇讲了从创建应用程序域到创建ISAPIRuntime实例的过程,本篇继续讲Asp.net处理第一次请求的必要的初始化过程. ISAPIRuntime分析 ISAPIRuntime在System.W ...
- u-boot中nandflash初始化流程分析(转)
u-boot中nandflash初始化流程分析(转) 原文地址http://zhuairlunjj.blog.163.com/blog/static/80050945201092011249136/ ...
- Spring MVC 处理一个请求的流程分析
Spring MVC是Spring系列框架中使用频率最高的部分.不管是Spring Boot还是传统的Spring项目,只要是Web项目都会使用到Spring MVC部分.因此程序员一定要熟练掌握MV ...
- Raid1源代码分析--初始化流程
初始化流程代码量比较少,也比较简单.主要是run函数.(我阅读的代码的linux内核版本是2.6.32.61) 四.初始化流程分析 run函数顾名思义,很简单这就是在RAID1开始运行时调用,进行一些 ...
- ASP.NET/MVC/Core的HTTP请求流程
ASP.NET HTTP管道(Pipeline)模型 1. 先讲一点,再深刻思考 一般我们都在写业务代码,优化页面,优化逻辑之间内徘徊.也许我们懂得HTTP,HTTPS的GET,POST,但是我们大部 ...
- 简述C#中IO的应用 RabbitMQ安装笔记 一次线上问题引发的对于C#中相等判断的思考 ef和mysql使用(一) ASP.NET/MVC/Core的HTTP请求流程
简述C#中IO的应用 在.NET Framework 中. System.IO 命名空间主要包含基于文件(和基于内存)的输入输出(I/O)服务的相关基础类库.和其他命名空间一样. System.I ...
- Asp.net MVC的Model Binder工作流程以及扩展方法(2) - Binder Attribute
上篇文章中分析了Custom Binder的弊端: 由于Custom Binder是和具体的类型相关,比如指定类型A由我们的Custom Binder解析,那么导致系统运行中的所有Action的访问参 ...
- Spring MVC启动流程分析
本文是Spring MVC系列博客的第一篇,后续会汇总成贴子. Spring MVC是Spring系列框架中使用频率最高的部分.不管是Spring Boot还是传统的Spring项目,只要是Web项目 ...
- ASP.NET MVC学前篇之请求流程
ASP.NET MVC学前篇之请求流程 请求流程描述 对于请求的流程,文章的重点是讲HttpApplication和HttpModule之间的关系,以及一个简单的示例实现.(HttpModule又是M ...
随机推荐
- 最简单的html5语言
什么是 HTML5? HTML5 是下一代 HTML 标准. 最小的HTML5文档 下面是一个简单的HTML5文档: <</span>!DOCTYPE html><< ...
- bootstrap基础
相信大多数后端开发人员的html,css并不是太好(主要说我).想要做一些网页效果,难度会比较大.看了下bootstrap这个前端框架,发现这个框架比较好的解决了网页效果制作中一般性问题.总的来说,b ...
- [原创]JS实现数据筛选(each)
做列表的时候,有时候需要按照某些条件进行查询,如班级,分组之类.但是又不想在重新从服务器获取数据,可以进行隐藏
- JavaScript ,Python,java,Go系列算法之选择排序
常见的内部排序算法有:插入排序.希尔排序.选择排序.冒泡排序.归并排序.快速排序.堆排序.基数排序等. 用一张图概括: 选择排序 选择排序是一种简单直观的排序算法,无论什么数据进去都是O(n2) ...
- 异步工作流控制-condCall
在JavaScript编程中,异步操作一直是一个问题,回调是一种深层次的嵌套处理方式,我们也可以把嵌套处理转为直线处理以简化异步处理.有过prolog和erlang编程了解的同学可能对模式匹配有深刻的 ...
- JSP servlet的配置与使用
1. servlet 的配置文件内容如下所示 <servlet> <description>This is the description of my J2EE com ...
- 多线程编程-- part 3 多线程同步->synchronized关键字
多线程同时访问一个资源,可以会产生不可预料的结果,所以为这个资源加锁,访问资源的第一个线程为其加锁后,其他线程便不能在使用那个资源,直到锁被解除. 举个例子: 存款1000元,能取出800的时候我就取 ...
- ftp服务器可以连接但不能传输数据(proftpd)
问题:在客户端连接FTP服务器(proftpd)时可以正常连接,但是无法正常传输数据 ftp> ls530 Please login with USER and PASSPassive mode ...
- 解决R语言临时文件目录的问题(tempdir、tempfile)
最近在调用SparkR的时候,当用copy_to函数将R中的数据框导入到Spark时,会在默认的tempdir()目录下(这里默认目录即为/tmp)产生巨大的临时文件, 严重影响R脚本的运行,最终一番 ...
- 为什么说上ERP找死?
长期以来,管理软件领域流行着这样一句话“不上ERP等死,上了ERP找死”.根据为十九年管理软件开发的经验来看,“不上ERP等死”这句话不敢苟同,但“上了ERP找死”这句话倒有些同感.上ERP虽然不一定 ...