`
raisun_1988
  • 浏览: 114280 次
  • 性别: Icon_minigender_1
  • 来自: 长沙
社区版块
存档分类
最新评论

MonoRail学习笔记四:MonoRail基本流程分析

    博客分类:
  • .Net
阅读更多

主要分析MonoRail框架内主要的几个类之间的调用关系,特别是分析一下当我们在浏览器中输入一个链接时,后台是怎么调用的,直至最后显示出页面的一个大体流程。
了解之后我们也可以知道哪一部分是可以由我们改造的。
基于Castle 1.0 RC3版本,和Castle 1.0稍微有点区别

首先看一下最简的一个web.config配置文件:

<?xmlversion="1.0"encoding="utf-8"?>
<configuration>
<configSections>
<sectionname="monorail"type="Castle.MonoRail.Framework.Configuration.MonoRailSectionHandler,Castle.MonoRail.Framework"/>
</configSections>

<monorail>
<controllers>
<assembly>TestSiteNVelocity</assembly>
</controllers>

<viewEnginesviewPathRoot="views">
<addtype="Castle.MonoRail.Framework.Views.NVelocity.NVelocityViewEngine,Castle.MonoRail.Framework.Views.NVelocity"xhtml="false"/>
</viewEngines>
</monorail>

<system.web>
<httpHandlers>
<addverb="*"path="*.rails"type="Castle.MonoRail.Framework.MonoRailHttpHandlerFactory,Castle.MonoRail.Framework"/>
</httpHandlers>
<httpModules>
<addname="monorail"type="Castle.MonoRail.Framework.EngineContextModule,Castle.MonoRail.Framework"/>
</httpModules>
</system.web>
</configuration>


可以看到在web.config中出现了三个类,下面我就按照程序执行的顺序来分析一下:
(以输入最简单的http://localhost:***/home/index.rails为例)

首先执行EngineContextModule类的Init方法:

publicvoidInit(HttpApplicationcontext)
{
if(context.Context.Error!=null)
{
thrownewException(
"AnexceptionhappenedonGlobalapplicationoronamodulethatrunbeforeMonoRail'smodule."+
"MonoRailwillnotbeinitializedandfurtherrequestsaregoingtofail."+
"Fixthecauseoftheerrorreportedbelow.",context.Context.Error);
}


lock(initLock)
{
CreateAndStartContainer(context);
}


context.BeginRequest
+=newEventHandler(OnStartMonoRailRequest);
context.AuthorizeRequest
+=newEventHandler(CreateControllerAndRunStartRequestFilters);

SubscribeToApplicationHooks(context);
}


在Init方法中会读取配置文件,根据配置文件生成相应的类,
如:XmlNodeList services = section.SelectNodes("services/service");
比如我们下面提到的DefaultUrlTokenizer类,当我们在web.config中配置了相应的类后,在解析Url时会使用我们自己配置的类,缺省情况下才使用DefaultUrlTokenizer

然后执行OnStartMonoRailRequest方法:
主要是把Url等信息解析好放入context中,context.Items[RailsContextKey] = newContext; 这里放入的是一个DefaultRailsEngineContext对象,这也是特别重要的一个对象,里面包含我们要调用的Controller类等信息。
在解析Url时默认用的是DefaultUrlTokenizer类,主要以下一段话:

string[]parts=rawUrl.Split('/');

if(parts.Length<2)
{
thrownewUrlTokenizerException("Urlsmallerthan2tokens");
}


action
=parts[parts.Length-1];

也就是当我们输入http://localhost:****/index.rails时会报错,必须有两个/的情况才允许的

接下来执行CreateControllerAndRunStartRequestFilters方法,这个方法最主要的就是生成Controller类:
return (Controller) Activator.CreateInstance(type);


然后进入MonoRailHttpHandlerFactory类的GetHandler方法
默认情况下调用以下语句:
mrHandler = new MonoRailHttpHandler(logger);

接着再进入MonoRailHttpHandler类的ProcessRequest方法
这里首先取得处理好的context对象:
IRailsEngineContext mrContext = EngineContextModule.ObtainRailsEngineContext(context);
通过mrContext对象最终调用动态方法:
method.Invoke(this, new object[0]);
也就是执行HomeController中的index方法,执行完之后,显示页面:

privatevoidProcessView()
{
if(controller._selectedViewName!=null)
{
viewEngineManager.Process(context,controller,controller._selectedViewName);
}

}

显示页面时就会调用我们web.config中配置的NVelocityViewEngine类去实际输出页面

小结:
从以上分析可以看出MonoRail的设计是很灵活的,很多思想可以借鉴的,
首先Module部分可以被替换,我们可以实现自己的Module,
其次viewEngines部分可以替换(这个很多人都知道了)
然后一些Services也可以动态替换,比如我前面提到的解析Url的类,可以用我们自己的方式解析,来实现不同的Controller和Url的映射关系。

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics