0 前言
AspNet MVC中比較重要的上下文,有如下:
- 核心的上下文有HttpContext(請求上下文),ControllerContext(控制器上下文)
- 過濾器有關有五個的上下文ActionExecutingContext,ActionExecutedContext,ResultExecutingContext,ResultExecutedContext,ExceptionContext
- 視圖相關的上下文ViewContext
這些上下文之間的關系如下圖所示

說明:
1、ControllerContext是對HttpContext的封裝
2、過濾器等filterContext上下文都是繼承自ControllerContext
3、ViewContext也是繼承自ControllerContext,同時封裝了對視圖的對象
由此可以看出,最基礎還是Aspnet的HttpContext上下文貫穿整個請求/應答的,而Mvc是將HttpContext進行再次封裝成ControllerContext。所以先看明白HttpContext與ControllerContext的來龍去脈即可大致了解這些上下文。
1 HttpContext的由來
先看看園里大叔的一張圖,如下所示。

大致的流程如下
- AppManagerAppDomainFactory類實現IAppManagerAppDomainFactory接口的Create方法,內部實現了創建AppDomain【HttpRuntime、HttpContext等都依附在AppDomain】、HostingEnvironment等一系列操作,並且得到一個ISAPIRuntime。
- 當IIS接受一個請求就可以通過上一步所得到的ISAPIRuntime的ProcessRequest進行處理請求。其間
①必須對WorkRequest針對不同的IIS版本進行包裝,從而創建得到ISAPIWorkerRequest實例對象
②HttpRuntime調用ProcessRequestNoDemand處理上面所得到的WorkRequest,並且通過ProcessRequestInternal 實例化化請求的上下文,如下代碼所示
HttpContext context = new HttpContext(wr/WorkRequest*/, false /* initResponseWriter */);
③HttpContext的構造函數內部也初始化HttpRequest以及HttpResponse
具體的內部細節,可以猛戳這里去看大叔深入剖析
2 ControllerContext
ControllerContext在ControllerBase的Initialize方法內部被實例化,ControllerBase作為基類,被后期控制器所繼承。ControllerContext也將作為其他的過濾器上下文的基類。
protected virtual void Initialize(RequestContext requestContext) {
ControllerContext = new ControllerContext(requestContext, this);
}
public RequestContext RequestContext {
get {
if (_requestContext == null) {
// still need explicit calls to constructors since the property getters are virtual and might return null
HttpContextBase httpContext = HttpContext ?? new EmptyHttpContext();
RouteData routeData = RouteData ?? new RouteData();
_requestContext = new RequestContext(httpContext, routeData);
}
return _requestContext;
}
set {
_requestContext = value;
}
}
3 過濾器上下文
過濾器采用AOP(面向切面編程),可以通過實現IActionFilter,IResultFilter,IExceptionFilter,IAuthorizationFilter接口,進行附加的過濾效果。這些接口的內部方法的參數有相對應的上下文,如IActionFilter內部含有ActionExecutingContext,ActionExecutedContext上下文,而且將ControllerActionInvoker的InvokeActionMethodWithFilters內部被實例化
public interface IActionFilter {
void OnActionExecuting(ActionExecutingContext filterContext);
void OnActionExecuted(ActionExecutedContext filterContext);
}
protected virtual ActionExecutedContext InvokeActionMethodWithFilters(ControllerContext controllerContext, IList<IActionFilter> filters, ActionDescriptor actionDescriptor, IDictionary<string, object> parameters) {
ActionExecutingContext preContext = new ActionExecutingContext(controllerContext, actionDescriptor, parameters);
//省略
}
4 視圖上下文
視圖上下文被實例化三個地方:ViewResultBase,HttpHelper、TemplateHelpers,該上下文更多的為渲染視圖提供數據支持。以ViewResultBase(繼承ActionResult並重寫ExecuteResult方法內部對ViewContext進行實例化)為例,如下
public override void ExecuteResult(ControllerContext context) {
if (context == null) {
throw new ArgumentNullException("context");
}
if (String.IsNullOrEmpty(ViewName)) {
ViewName = context.RouteData.GetRequiredString("action");
}
ViewEngineResult result = null;
if (View == null) {
result = FindView(context);
View = result.View;
}
TextWriter writer = context.HttpContext.Response.Output;
ViewContext viewContext = new ViewContext(context, View, ViewData, TempData, writer);
View.Render(viewContext, writer);
if (result != null) {
result.ViewEngine.ReleaseView(context, View);
}
}
致此,基本介紹了MVC內部的上下文內容。如果覺得不錯請點贊下,有誤的話請指出,謝謝!
