不使用特殊方法来处理请求生命周期事件
HttpApplication 类是全局应用类的基类,定义了可以直接使用的一般 C# 事件。那么使用标准 C# 事件还是特殊方法那就是个人偏好的问题了,如果喜欢,也可以将这两种方式混合起来使用。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using System.Web.Routing; namespace SimpleApp
{
public class MvcApplication : System.Web.HttpApplication
{
public MvcApplication()
{
BeginRequest += (src, args) => RecordEvent("BeginRequest");
AuthenticateRequest += (src, args) => RecordEvent("AuthenticateRequest");
PostAuthenticateRequest += (src, args) => RecordEvent("PostAuthenticateRequest");
} protected void Application_Start()
{
AreaRegistration.RegisterAllAreas();
RouteConfig.RegisterRoutes(RouteTable.Routes);
} private void RecordEvent(string name)
{
List<string> eventList = Application["events"] as List<string>;
if (eventList == null)
{
Application["events"] = eventList = new List<string>();
}
eventList.Add(name);
}
}
}
我们已经在 MvcApplication 类中添加了一个构造器,并为其中三个请求生命周期事件建立事件处理器。而这三个事件我都使用了 lambda 表达式来调用 RecordEvent 方法,存储事件的名称供 controller 来读取,与之前的例子一样。
提示:没有标准的 C# 事件来代替 Application_Start 和 Application_End 方法。我们仅能够通过特殊方法来接收这两个通知。
使用单个方法处理多个事件
如果我们想要使用一个方法来处理多个生命周期事件而不依赖 lambda 表达式,那么 System.Web.HttpContext 类中定义的两个属性可以帮助我们。HttpContext 类中提供了当前请求和应用状态的详细信息,我们会在后面详细介绍。然而目前,我们只会介绍与处理生命周期事件的两个属性。
名称 | 描述 |
CurrentNotification | 这个属性会使用 System.Web.RequestNotification 枚举中的值来指示当前应用事件。 |
IsPostNotification | 如果 CurrentNotification 属性返回的当前应用事件名称是 Post<Name> 的变体,那么这个属性就会返回 true。 |
这两个属性有一点奇怪,因为两者都需要计算出哪个事件正在被处理。CurrentNotification 属性返回一个 RequestNotification 枚举值,当中定义了 HttpApplication 事件的子集。IsPostNotification 属性值是根据当前触发的事件是否是一个像 AcquireRequestState 的事件还是与其配对的 PostAcquireRequestState 事件计算出来的。在 HttpApplication 类中,可以通过 Context 属性获取到 HttpContext 对象。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using System.Web.Routing; namespace SimpleApp
{
public class MvcApplication : System.Web.HttpApplication
{
public MvcApplication()
{
BeginRequest += RecordEvent;
AuthenticateRequest += RecordEvent;
PostAuthenticateRequest += RecordEvent;
} protected void Application_Start()
{
AreaRegistration.RegisterAllAreas();
RouteConfig.RegisterRoutes(RouteTable.Routes);
} private void RecordEvent(object src, EventArgs args)
{
List<string> eventList = Application["events"] as List<string>;
if (eventList == null)
{
Application["events"] = eventList = new List<string>();
}
string name = Context.CurrentNotification.ToString();
if (Context.IsPostNotification)
{
name = "Post" + name;
}
eventList.Add(name);
}
}
}
我改变了 RecordEvent 方法的签名,因此采用了标准的事件处理器签名:一个对象代表的是事件的源,EventArgs 对象描述的就是事件。我没有使用这两个参数提供的值,相反,我使用了 Context.CurrentNotification 属性和 Context.IsPostNotification 属性提供的相关信息。
我不明白微软为什么以这样的一个方式来实现事件,但是如果你不想使用特殊方法或者 lambda 表达式的话,那你就必须使用这种方式。注意到,在上面的代码中,我在 Context.CurrentNotification 属性上使用了 ToString 方法,这是必须的,因为 CurrentNotification 属性返回的是一个 System.Web.RequestNotification 枚举值。详见下表:
表 1 – RequestNotification 枚举值
值 | 描述 |
BeginRequest | 对应到 BeginRequest 事件 |
AuthenticateRequest | 对应到 AuthenticateRequest 和 PostAuthenticateRequest 事件 |
AuthorizeRequest | 对应到 AuthorizeRequest 事件 |
ResolveRequestCache | 对应到 ResolveRequestCache 和 PostResolveRequestCache 事件 |
MapRequestHandler | 对应到 MapRequestHandler和 PostMapRequestHandler 事件 |
AcquireRequestState | 对应到 AcquireRequestState 和 PostRequestState 事件 |
PreExecuteRequestHandler | 对应到 PreExecuteRequestHandler 事件 |
ExecuteRequestHandler | 对应到 ExecuteRequestHandler 事件 |
ReleaseRequestState | 对应到 ReleaseRequestState 和 PostReleaseRequestState 事件 |
UpdateRequestCache | 对应到 UpdateRequestCahce 事件 |
LogRequest | 对应到 LogRequest 事件 |
EndRequest | 对应到 EndRequest 事件 |
SendResponse | 指示响应正在被发送——不完全对应到 PreSendRequestHeaders 和 PreSendRequestContent 事件 |
[根据 Adam Freeman – Pro ASP.NET MVC 5 Platform 选译]