码迷,mamicode.com
首页 > Web开发 > 详细

ASP.NET MVC : Action过滤器(Filtering)

时间:2015-09-09 16:25:32      阅读:310      评论:0      收藏:0      [点我收藏+]

标签:

http://www.cnblogs.com/QLeelulu/archive/2008/03/21/1117092.html

ASP.NET MVC : Action过滤器(Filtering)

相关文章:

  1. ASP.NET MVC URL Routing 学习
  2. AP.NET MVC : 控制器 和 控制器Actions
  3. ASP.NET MVC 学习: 视图  

有时候你想在调用action方法之前或者action方法之后处理一些逻辑,为了支持这个,ASP.NET MVC允许你创建action过滤器。Action过滤器是自定义的Attributes,用来标记添加Action方法之前或者Action方法之后的行为到控制器类中的Action方法中。

一些可能用到Action过滤器的地方有:

  • 日志
  • 身份验证和授权 - 限制用户的访问
  • 输出缓存 - 保存一个Action的结果
  • 网络爬虫的过滤
  • 本地化
  • 动态Action - 将一个Action注入到控制器中

实现一个Action过滤器

Action过滤器是通过继承ActionFilterAttribute类来实现的一个Attribute类。ActionFilterAttribute 是一个抽象类,提供了两个virtual的方法给我们重写,OnActionExecutingOnActionExecuted

ASP.NET MVC 框架会在调用Action方法之前调用你Action过滤器中的OnActionExecuting方法,在之后调用Action过滤器中的OnActionExecuted方法。当然在创建Action过滤器的时候你不需两个方法都实现。

下面的示例是在调用Action方法之前和之后的日志跟踪:

技术分享
技术分享public class LoggingFilterAttribute : ActionFilterAttribute
技术分享{
技术分享    public override void OnActionExecuting(FilterExecutingContext 
技术分享        filterContext)
技术分享    {
技术分享        filterContext.HttpContext.Trace.Write("Starting: " + 
技术分享        filterContext.ActionMethod.Name);
技术分享    } 
技术分享
技术分享    public override void OnActionExecuted(FilterExecutedContext 
技术分享        filterContext)
技术分享    {
技术分享        if (filterContext.Exception != null)
技术分享        {
技术分享            filterContext.HttpContext.Trace.Write("Exception thrown");
技术分享        }
技术分享    }
技术分享}

Action Filter Context

OnActionExecuting方法有一个类型为FilterExecutingContext的参数,而OnActionExecuted方法有一个相应的类型为FilterExcutedContext的参数。两个Context类都是继承自FilterContext类,而FilterContext类继承自ControllerContext类并包含一个ActionMethod属性。你可以使用ActionMethod属性来坚定这个Action过滤器是应用到哪个Action方法上的。

FilterExecutingContext类包含一个 Cancel 的属性,允许你取消当前的Action。

FilterExcutedContext 类包含一个Exception属性和一个ExceptionHandled属性。如果Exception属性为null,则没有异常在action stack中,表明Action方法运行并没有发生错误。如果Exception属性不为null,则过滤器知道该怎么处理,过滤器处理完异常后会发出已经处理完的信号,然后将ExceptionHandled属性设为true。就算ExceptionHandled属性为true,堆栈中添加到其他Action方法的OnActionExcetued方法将会照常被调用,这种场景就如就算一个异常被处理了,日志记录filter一样照常执行。

用过滤器特性(Attribute)来标记一个Action方法

你可以将过滤器应用到任何一个你喜欢的Action方法上。下面的示例演示一个控制器中包含的用Action过滤器Attribute标记的Action方法。

技术分享public class HomeController : Controller
技术分享{
技术分享    [LoggingFilter]
技术分享    public void Index()
技术分享    {
技术分享        RenderView("Index");
技术分享    } 
技术分享
技术分享    [LoggingFilter]
技术分享    public void About()
技术分享    {
技术分享        RenderView("About");
技术分享    } 
技术分享
技术分享    [LoggingFilter]
技术分享    public void ClickMe()
技术分享    {
技术分享        HttpContext.Trace.Write("Button was clicked.");
技术分享        InvokeAction("Index");
技术分享    }
技术分享}

 

实现一个控制器范围的Action过滤器

ASP.NET MVC 控制器(Controller)类定义的OnActionExecuting 和 OnActionExcuted 方法你可以重写。当你重写一个或者这两个方法的时候,你实际上定义了一个将会应用到该控制器类中所有的Action方法的Action过滤器。严格来说,这个方法没有构成一个Action过滤器,但不管怎样,她们提供的功能是相似的。
在下面的示例中,控制器级别的OnActionExecuting和OnActionExecuted方法应用到控制器中所有的Action方法中:

技术分享
技术分享public class HomeController : Controller
技术分享{
技术分享    public void Index()
技术分享    {
技术分享        RenderView("Index");
技术分享    }
技术分享
技术分享    public void About()
技术分享    {
技术分享        RenderView("About");
技术分享    }
技术分享
技术分享    public void ClickMe()
技术分享    {
技术分享        HttpContext.Trace.Write("Button was clicked.");
技术分享        InvokeAction("Index");
技术分享    }
技术分享
技术分享    protected override void OnActionExecuting(FilterExecutingContext 
技术分享        filterContext)
技术分享    {
技术分享        filterContext.HttpContext.Trace.Write("Starting: " + 
技术分享            filterContext.ActionMethod.Name);
技术分享    }
技术分享
技术分享    protected override void OnActionExecuted(FilterExecutedContext 
技术分享        filterContext)
技术分享    {
技术分享        if (filterContext.Exception != null)
技术分享        {
技术分享            filterContext.HttpContext.Trace.Write("Exception thrown");
技术分享        }
技术分享    }
技术分享}


Action过滤器的作用范围

除了用Action过滤器标记一个Action方法外,你也可以用来标记一个完成的控制器类。如果这样的话,这个Action过滤器将会应用到该控制器的所有Action方法上。

另外,如果你的控制器类继承自别的控制器类,而基控制器类可能有它自己的Action过滤器Attributes。如果你在子类中重写了基控制器类的Action方法,则子类的该Action方法也会有它自己的从基类继承而来的Action过滤器Attributes。

Action过滤器的执行顺序

每一个Action过滤器都有一个 Order 属性,用来决定Action过滤器在该范围内的执行顺序。Order属性必需是0(默认值)或者更大的整数值。省略Order属性则会给该过滤器的Order值为 -1, 表明为指明顺序。任何一个在同一范围的Action过滤器Order设为 -1 的都将按不确定的顺序执行,单在此之前过滤器有一个特定的顺序(注:下面会说到).

当设置Order属性的值的时候,必需指定一个唯一的值。如果两个或者更多的Action过滤器具有相同的Order属性值,将会抛出一个异常。

来看一个示例:

[Filter1(Order = 2)]
[Filter2(Order = 3)]
[Filter3(Order = 1)]
public void Index()
{
    RenderView("Index");
}

Filter的执行顺序为:Filter3 => Filter1 => Filter2.

在同一范围,Action过滤器按下面的顺序执行:

  • OnActionExecuting virtual method of the controller.
  • OnActionExecuting method of any filters that are applied to the current controller, in this order:
    • Base class

    • Any derived class

  • OnActionExecuting method of filters that are applied to the action method, in this order:
    • Base class

    • Derived class

  • Action method
  • OnActionExecuted method of filters that are applied to the action method, in this order:
    • Derived class

    • Base class

  • OnActionExecuted method of filters that are applied to the current controller, in this order:
    • Derived class

    • Base class

  • OnActionExecuted virtual method of the controller.

可以看到OnActionExecuted 和 OnActionExecuting 的执行顺序是反过来的。

Action过滤器执行顺序的示例

下面的示例演示了一个包含两个Action过滤器的MVC程序。DebugFilter 过滤器写trace信息,而ThrowExceptionFilter 过滤器引起一个异常。这个程序也包含一个基控制器和一个子控制器和一个视图。

技术分享
技术分享public class DebugFilterAttribute : ActionFilterAttribute
技术分享{
技术分享    public static void Reset()
技术分享    {
技术分享        counter = 1;
技术分享        indent = 0;
技术分享        incrementIndex = true;
技术分享    }
技术分享
技术分享    public static int Counter
技术分享    {
技术分享        get
技术分享        {
技术分享            return counter++;
技术分享        }
技术分享    }
技术分享    
技术分享    static int counter
技术分享    {
技术分享        get
技术分享        {
技术分享            return (int)(HttpContext.Current.Items["counter"] ?? 1);
技术分享        }
技术分享        set
技术分享        {
技术分享            HttpContext.Current.Items["counter"] = value;
技术分享        }
技术分享    }
技术分享
技术分享    static int indent
技术分享    {
技术分享        get
技术分享        {
技术分享            return (int)(HttpContext.Current.Items["indent"] ?? 1);
技术分享        }
技术分享        set
技术分享        {
技术分享            HttpContext.Current.Items["indent"] = value;
技术分享        }
技术分享    }
技术分享
技术分享    static bool incrementIndex
技术分享    {
技术分享        get
技术分享        {
技术分享            return (bool)(HttpContext.Current.Items["incrementIndex"] ?? true);
技术分享        }
技术分享        set
技术分享        {
技术分享            HttpContext.Current.Items["incrementIndex"] = value;
技术分享        }
技术分享    }
技术分享
技术分享    public string Message { get; set; }
技术分享
技术分享    public int Id { get; set; }
技术分享
技术分享    public override void OnActionExecuting(FilterExecutingContext 
技术分享         filterContext)
技术分享    {
技术分享        HttpContext.Current.Trace.Write(
技术分享            "Action Filter", 
技术分享            string.Format("{0}: {3}(PRE) DebugFilter.OnActionExecuting"
技术分享            + " - Order={1} Message=‘{2}‘", 
技术分享            Counter, 
技术分享            this.Order, 
技术分享            this.Message, 
技术分享            Indent));
技术分享    }
技术分享
技术分享    public static string Indent
技术分享    {
技术分享        get
技术分享        {
技术分享            indent += (incrementIndex ? 1 : -1);
技术分享            string indentText = string.Empty;
技术分享            for (int i = 0; i < indent; i++)
技术分享                indentText += " ";
技术分享            return indentText;
技术分享        }
技术分享    }
技术分享
技术分享    public static void StartDecrement()
技术分享    {
技术分享        incrementIndex = false;
技术分享    }
技术分享}

下面的Action过滤器在调用的时候将会抛出一个异常:
技术分享public class ThrowExceptionFilter : DebugFilterAttribute
技术分享{
技术分享    public override void 
技术分享        OnActionExecuting(
技术分享            System.Web.Mvc.FilterExecutingContext filterContext)
技术分享    {
技术分享        base.OnActionExecuting(filterContext);
技术分享        DebugFilterAttribute.StartDecrement();
技术分享        throw new InvalidOperationException(
技术分享            "Exception thrown by the filter");
技术分享    }
技术分享}

下面的是定义一个应用了Action过滤器的基控制器类:
技术分享[DebugFilter(Message = "(CONTROLLER) MyBaseController", Order = 1)]
技术分享[DebugFilter(Message = "(CONTROLLER) MyBaseController", Order=2)]
技术分享public class MyBaseController : Controller
技术分享{
技术分享    [ControllerAction]
技术分享    [DebugFilter(Message = "(ACTION) MyBaseController.Index()", Order=2)]
技术分享    [DebugFilter(Message = "(ACTION) MyBaseController.Index()", Order=1)]
技术分享    public virtual void Index()
技术分享    {
技术分享        RenderView("Index");
技术分享    }
技术分享}

下面的是子控制器类,注意这里重写了OnActionExecuting 方法:
技术分享
技术分享[DebugFilter(Message = "(CONTROLLER) MyDerivedController", Order = 2)]
技术分享[DebugFilter(Message = "(CONTROLLER) MyDerivedController", Order=1)]
技术分享public class MyDerivedController : MyBaseController
技术分享{
技术分享    [ControllerAction]
技术分享    [DebugFilter(Message = "(ACTION) MyDerivedController.Index()", Order=1)]
技术分享    [DebugFilter(Message = "(ACTION) MyDerivedController.Index()", Order = 2)]
技术分享    public override void Index()
技术分享    {
技术分享        base.Index();
技术分享        HttpContext.Trace.Write(
技术分享            "Action Filter", 
技术分享            string.Format("{0}: {1}(ACTION) "
技术分享            + "MyDerivedController.Index()", 
技术分享            DebugFilterAttribute.Counter, 
技术分享            DebugFilterAttribute.Indent));
技术分享        DebugFilterAttribute.StartDecrement();
技术分享    }
技术分享
技术分享    protected override void OnActionExecuting(FilterExecutingContext filterContext)
技术分享    {
技术分享        //The Beginning
技术分享        DebugFilterAttribute.Reset();
技术分享
技术分享        HttpContext.Trace.Write(
技术分享            "Action Filter", 
技术分享            string.Format("{0}: {1}(PRE) "
技术分享            + "MyDerviedController.OnActionExecuting VIRTUAL METHOD",
技术分享            DebugFilterAttribute.Counter, 
技术分享            DebugFilterAttribute.Indent));
技术分享    }
技术分享}

下面的是显示trace的视图页:
技术分享
技术分享<%@ Page Language="C#" CodeBehind="Index.aspx.cs" Inherits="ActionFilterTests.Views.MyDerived.Index" Trace="true" %>
技术分享
技术分享<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
技术分享
技术分享<html xmlns="http://www.w3.org/1999/xhtml" >
技术分享<head runat="server">
技术分享    <title>Attribute Filter Test</title>
技术分享</head>
技术分享<body>
技术分享    <form id="form1" runat="server">
技术分享    <div>
技术分享        Hello World
技术分享    </div>
技术分享    </form>
技术分享</body>
技术分享</html>

下面是程序运行时的输出结果:
Action Filter 1:  (PRE) MyDerviedController.OnActionExecuting VIRTUAL METHOD   
Action Filter 2:   (PRE) DebugFilter.OnActionExecuting - Order=1 Message=‘(CONTROLLER) MyBaseController
Action Filter 3:    (PRE) DebugFilter.OnActionExecuting - Order=2 Message=‘(CONTROLLER) MyBaseController
Action Filter 4:     (PRE) DebugFilter.OnActionExecuting - Order=1 Message=‘(CONTROLLER) MyDerivedController
Action Filter 5:      (PRE) DebugFilter.OnActionExecuting - Order=2 Message=‘(CONTROLLER) MyDerivedController
Action Filter 6:       (PRE) DebugFilter.OnActionExecuting - Order=1 Message=‘(ACTION) MyDerivedController.Index()
Action Filter 7:        (PRE) DebugFilter.OnActionExecuting - Order=2 Message=‘(ACTION) MyDerivedController.Index()
Action Filter 8:         (ACTION) MyDerivedController.Index()

注:本文参考自http://quickstarts.asp.net/3-5-extensions/mvc/ActionFiltering.aspx

ASP.NET MVC : Action过滤器(Filtering)

标签:

原文地址:http://www.cnblogs.com/zkwarrior/p/4794832.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!