溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Asp.Net MVC過濾器的示例分析

發(fā)布時間:2021-07-27 12:31:07 來源:億速云 閱讀:116 作者:小新 欄目:編程語言

小編給大家分享一下Asp.Net MVC過濾器的示例分析,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

一、過濾器簡介

1.1、理解什么是過濾器

1、過濾器(Filters)就是向請求處理管道中注入額外的邏輯。提供了一個簡單而優(yōu)雅的方式來實(shí)現(xiàn)橫切關(guān)注點(diǎn)。

2、所謂的過濾器(Filters),MVC框架里面的過濾器完全不同于ASP.NET平臺里面的Request.Filters和Response.Filter對象,它們主要是實(shí)現(xiàn)請求和響應(yīng)流的傳輸。通常我們所說的過濾器是指MVC框架里面的過濾器。

3、過濾器可以注入一些代碼邏輯到請求處理管道中,是基于C#的Attribute的實(shí)現(xiàn)。當(dāng)負(fù)責(zé)調(diào)用Action的類ControllerActionInvoker在調(diào)用執(zhí)行Action的時候會檢查Action上面的Attribute并查看這些Attribute是否實(shí)現(xiàn)了指定的接口,以便進(jìn)行額外的代碼注入處理

1.2、理解為什么要使用過濾器

假設(shè)你做了一個小項(xiàng)目,其中某個功能是操作管理用戶信息模塊,有這樣一個需求,對用戶信息管理必須是已通過認(rèn)證的用戶才能操作,我們可以在每一個Action方法里面檢查認(rèn)證請求,如下所示:

using MvcFilterDmo.Core;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using System.Web.Security;

namespace MvcFilterDmo.Controllers
{
  public class HomeController : Controller
  {
    public ActionResult Index()
    {
      if (!Request.IsAuthenticated)
      {
        FormsAuthentication.RedirectToLoginPage();
      }
      //操作部分...
      return View();
    }
    public ActionResult Insert()
    {
      if (!Request.IsAuthenticated)
      {
        FormsAuthentication.RedirectToLoginPage();
      }
      //操作部分...
      return View();
    }
    public ActionResult Update()
    {
      if (!Request.IsAuthenticated)
      {
        FormsAuthentication.RedirectToLoginPage();
      }
      //操作部分...
      return View();
    }
    public ActionResult Delete()
    {
      if (!Request.IsAuthenticated)
      {
        FormsAuthentication.RedirectToLoginPage();
      }
      //操作部分...
      return View();
    }
    //其他Action操作方法
    //...
  }
}

通過上面的代碼,可以發(fā)現(xiàn)使用這種方式檢查請求認(rèn)證有許多重復(fù)的地方,這也就是為什么要使用過濾器的原因,使用過濾器可以實(shí)現(xiàn)相同的效果。如下所示:

using MvcFilterDmo.Core;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using System.Web.Security;

namespace MvcFilterDmo.Controllers
{
  [Authorize]
  public class HomeController : Controller
  {
    public ActionResult Index()
    {
      //操作部分...
      return View();
    }
    public ActionResult Insert()
    {
      //操作部分...
      return View();
    }
    public ActionResult Edit()
    { 
      //操作部分...
      return View();
    }
    public ActionResult Delete()
    {
      //操作部分...
      return View();
    }
    //其他Action操作方法
    //...
  }
}

過濾器是.NET里面的特性(Attributes),它提供了添加到請求處理管道的額外方法。這里使用Authorize過濾器可以實(shí)現(xiàn)同樣的效果,不過代碼就顯然比之前更加簡潔優(yōu)雅。

二、過濾器的使用

2.1、基本類型的過濾器

過濾器實(shí)現(xiàn)的機(jī)制:在MVC框架調(diào)用一個Action之前,它會檢查方法的定義中是否實(shí)現(xiàn)了特性(Attributes),如果實(shí)現(xiàn)的話,那么在請求處理管道適當(dāng)?shù)奈恢?,該特性定義的方法會被調(diào)用。

Asp.Net MVC過濾器的示例分析

ActionFilterAttribute類既實(shí)現(xiàn)了IactionFilter接口,也實(shí)現(xiàn)IResultFilter接口。這是一個抽象類,它要求你必須提供一個實(shí)現(xiàn)。AuthorizeAttribute和HandleErrorAttribute類,則包含了一些有用的特性,并且可以不必創(chuàng)建派生類進(jìn)行使用。

2.2、過濾器的應(yīng)用、應(yīng)用方式以及執(zhí)行順序

應(yīng)用: 過濾器可以被應(yīng)用到控制器上也可以用到Action方法上,應(yīng)用到控制上時,表示所有的Action方法都有了這個過濾器,并且可以混合使用,或多次使用,如下所示:

[A] //表示所有的Action方法都會應(yīng)用A過濾器
Public class DemoController:Controller
{
  [B]//B,C過濾器只作用于此Action方法,但它也會有A過濾器的應(yīng)用效果
  [C]
  Public ActionResult Index()
  {
     //操作部分...
     return View();
  } 
}

應(yīng)用方式:特性的方式,如上面代碼所示。

執(zhí)行順序:相同類型過濾器,執(zhí)行順序靠近方法的先執(zhí)行,不同類型的過濾器一般執(zhí)行順序?yàn)椤綼uthorize--->action--->actionResult】至于異常過濾器不分先后,只要拋出異常時就會執(zhí)行異常過濾器。如果要調(diào)整執(zhí)行順序,可以通過調(diào)整Order方法值大小來控制執(zhí)行順序,值越小,越先執(zhí)行。下圖是Action/Result過濾器應(yīng)用的執(zhí)行順序圖

Asp.Net MVC過濾器的示例分析

(1)、相同類型過濾器應(yīng)用示例:兩個自定義Action過濾器MyFirstFilter,MyThreeFilter應(yīng)用到同一個Action方法Index上。

Three控制器代碼如下:

Asp.Net MVC過濾器的示例分析

MyFirstFilter 代碼如下:

Asp.Net MVC過濾器的示例分析

MyThreeFilter代碼如下:

Asp.Net MVC過濾器的示例分析

運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

(2)、不同類型過濾器應(yīng)用示例:有一個自定義Action過濾器MyFirstFilter,有一個自定義Result過濾器MySecondFilter,應(yīng)用到同一個Action方法Index上。

Three控制器代碼如下:

Asp.Net MVC過濾器的示例分析

MyFirstFilter 代碼如下:

Asp.Net MVC過濾器的示例分析

MySecondFilter代碼如下:

Asp.Net MVC過濾器的示例分析

運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

看完上面的解釋,可能你現(xiàn)在對這些過濾器的執(zhí)行順序,以及如何自定義過濾器還不明白,不要緊,下面我們會逐一介紹這幾個基本的過濾器的使用,以及如何自定義過濾器。

2.3、使用授權(quán)過濾器

所有實(shí)現(xiàn)了IAuthorizationFilter接口的都可以稱之為授權(quán)過濾器:其定義如下:

 public interface IAuthorizationFilter
    {
       void OnAuthorization(AuthorizationContext filterContext);
    }

由于MVC框架系統(tǒng)自帶的AuthorizeAttribute實(shí)現(xiàn)有一些突出的功能,而這種牽涉到安全的代碼一定要謹(jǐn)慎的編寫,所以一般我們不會直接實(shí)現(xiàn)這個接口,而是去繼承AuthorizeAttribute這個類,并重寫其AuthorizeCore方法,簽名為: bool AuthorizeCore(HttpContextBase httpContext) 而處理授權(quán)失敗的時候,可以重寫其HandleUnauthorizedRequest方法,其簽名為: void HandleUnauthorizedRequest(AuthorizationContext context) 。注意:驗(yàn)證與授權(quán)是兩回事,驗(yàn)證發(fā)生在授權(quán)之前。

默認(rèn)的授權(quán)過濾器已經(jīng)有了驗(yàn)證的功能,其驗(yàn)證的機(jī)理是利用Asp.net平臺自帶的驗(yàn)證機(jī)制,如表單驗(yàn)證和Windows驗(yàn)證。除了驗(yàn)證功能,它本身還有授權(quán)的功能。授權(quán)過濾器是所有過濾器中最早運(yùn)行的。

經(jīng)過Route到達(dá)了控制器的時候,在調(diào)用Action之前,MVC框架會檢測在相關(guān)的Action上是否有授權(quán)過濾器,如果有會調(diào)用OnAuthorization方法,如果此方法批準(zhǔn)了請求,才會調(diào)用相應(yīng)的Action。

使用授權(quán)過濾器幾種情況如下:

1.直接在Action上或者控制器上加Authorize,表示啟用了驗(yàn)證,但不牽涉到授權(quán)。

2.添加Authorize(Users=“a,b”)],表示啟用了驗(yàn)證,并且也啟用了授權(quán),只有a或者b用戶能訪問此控制器。

3.當(dāng)添加Authorize(Roles=“admin,Member”)]時的步驟如下:

---利用asp.net自帶的角色提供者,或者實(shí)現(xiàn)自己的角色提供者,實(shí)現(xiàn)自己的角色提供者時,只需要集成RoleProvider類型,并實(shí)現(xiàn)其中的所有方法或部分方法,最好實(shí)現(xiàn)所有方法。

---在Web程序的根目錄的Web.config文件中配置角色管理者。

---在適當(dāng)?shù)腁ction中利用Roles類型來訪問自己創(chuàng)建的RoleProvider中的相關(guān)方法。

使用內(nèi)置的授權(quán)過濾器

MVC框架內(nèi)置的授權(quán)過濾器AuthorizeAttribute,它允許我們使用這個類的兩個公共屬性來指定授權(quán)策略,如下所示:

Asp.Net MVC過濾器的示例分析

Asp.Net MVC過濾器的示例分析

Users和Roles兩者是并且的關(guān)系,例如Users=“a,b,c”,Roles=“admin”,表示用戶是a,b,c 其中一個并且是Admin角色才能訪問。

創(chuàng)建自定義的授權(quán)過濾器

方式一:直接實(shí)現(xiàn)IAuthorizationFilter接口,但不推薦這樣做,因?yàn)闋可娴桨踩矫娴拇a。

方式二:繼承AuthorizeAttribute這個類,并重寫其AuthorizeCore方法,簽名為: bool AuthorizeCore(HttpContextBase httpContext),代碼如下所示:

public class MyAuthorizeAttribute : AuthorizeAttribute
  {
    private string[] allowedUsers;
    public MyAuthorizeAttribute(params string[] users)
    {
      allowedUsers = new string[] { "admin", "user1", "xf" };
    }

    protected override bool AuthorizeCore(HttpContextBase httpContext)
    {
      return httpContext.Request.IsAuthenticated &&allowedUsers.Contains(httpContext.User.Identity.Name, 
        StringComparer.InvariantCultureIgnoreCase);
    }
  }

2.4、使用動作過濾器

動作過濾器是可以以用于任何目的的多用途過濾器,創(chuàng)建自定義動作過濾器需要實(shí)現(xiàn)IActionFilter接口,該接口代碼如下所示:

Asp.Net MVC過濾器的示例分析

該接口定義了兩個方法,MVC框架在調(diào)用動作方法之前,會調(diào)用OnActionExecting方法。在調(diào)用動作方法之后,則會調(diào)用OnActionExecuted方法。

實(shí)現(xiàn)OnActionExecting方法

參數(shù)ActionExecutingContext對象繼承于ControllerContext,其中的2個屬性:

ActionDescriptor:提供了關(guān)于Action方法的相關(guān)信息

Result:類型為ActionResult,通過給這個屬性設(shè)置一個非null的值就可以取消這個請求。

我們可以用過濾器來取消一個請求,通過設(shè)置Result屬性即可。代碼如下所示:

public class MyActionFilterAttribute : FilterAttribute, IActionFilter
  {
    public void OnActionExecuting(ActionExecutingContext filterContext)
    {
      if(filterContext.HttpContext.Request.IsLocal)
      {
        filterContext.Result = new HttpNotFoundResult();
      }
    }
    public void OnActionExecuted(ActionExecutedContext filterContext)
    {
      //未做實(shí)現(xiàn)
    }
  }

這個例子通過用OnActionExecuting方法檢查請求是否來自本地機(jī)器,如果是,編隊用戶返回一個“404”未找到的響應(yīng)。運(yùn)行結(jié)果如下圖:

Asp.Net MVC過濾器的示例分析

實(shí)現(xiàn)OnActionExecuted方法

我們也可以通過OnActionExecuted方法來執(zhí)行一些跨越動作方法的任務(wù),下面這個例子是計算動作方法運(yùn)行的時間,代碼如下:

public class MyActionFilterAttribute : FilterAttribute, IActionFilter
  {
    private Stopwatch timer;
    public void OnActionExecuting(ActionExecutingContext filterContext)
    {
      timer = Stopwatch.StartNew();
    }
    public void OnActionExecuted(ActionExecutedContext filterContext)
    {
      timer.Stop();
      if (filterContext.Exception == null)
      {
        filterContext.HttpContext.Response.Write(
          string.Format("動作方法延遲的時間: {0}",
            timer.Elapsed.TotalSeconds));
      }
    }
  }
}

我們將自定義的動作過濾器MyActionFilter應(yīng)用到HomeController的Index方法上,運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

2.5、使用結(jié)果過濾器

結(jié)果過濾器是多用途的過濾器,他會對動作方法所產(chǎn)生結(jié)果進(jìn)行操作,結(jié)果過濾器實(shí)現(xiàn)IResultFilter接口,創(chuàng)建自定義結(jié)果過濾器需要現(xiàn)IResultFilter接口,該接口代碼如下所示:

Asp.Net MVC過濾器的示例分析

當(dāng)結(jié)果過濾器運(yùn)用于一個動作方法時,會在動作方法返回動作結(jié)果之前,調(diào)用OnResultExecuting方法,在返回動作結(jié)果之后,會調(diào)用OnResultExecuted方法。下面這個例子是計算動作方法返回結(jié)果運(yùn)行的時間,代碼如下:

public class MyResultFilterAttribute : FilterAttribute, IResultFilter
  {
    private Stopwatch timer;
    public void OnResultExecuting(ResultExecutingContext filterContext)
    {
      timer = Stopwatch.StartNew();
    }
    public void OnResultExecuted(ResultExecutedContext filterContext)
    {
      timer.Stop();
      filterContext.HttpContext.Response.Write(string.Format("結(jié)果執(zhí)行延遲時間: {0}", timer.Elapsed.TotalSeconds));
    }
}

我們將自定義的結(jié)果過濾器MyResultFilter應(yīng)用到HomeController的Index方法上,運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

需要注意的是:動作過濾器是運(yùn)行在頁面輸出之前,結(jié)果過濾器是運(yùn)行在頁面輸出之后。

2.6、使用異常過濾器

異常過濾器只有在調(diào)用一個動作方法而拋出未處理的異常才會運(yùn)行,這種異常來自以下位置:

A、另一種過濾器(授權(quán)、動作、或結(jié)果過濾器)。

B、動作方法本身。

C、當(dāng)動作結(jié)果被執(zhí)行時。

使用內(nèi)置的異常過濾器

HandleErrorAttribute(處理程序錯誤特性),它是MVC內(nèi)嵌的異常過濾器,有以下3個重要的屬性:

1.ExceptionType:類型為Type,表示希望被此過濾器處理的異常類型,包括其子類型,默認(rèn)值為System.Exception

2.View:類型為string,表示此過濾器呈遞的視圖頁面,默認(rèn)值為Error

3.Master:呈遞的視圖頁的母板頁,如果不指定,視圖會用其默認(rèn)的母版頁

內(nèi)嵌的HandleErrorException只有在配置文件Web.config中配置的CustomError 的mode設(shè)置為on的時候才生效(其默認(rèn)模式為RemoteOnly),如下圖所示:

Asp.Net MVC過濾器的示例分析

此過濾器還會給視圖傳遞一個HandleErrorInfo類型的對象給視圖,以便視圖可以顯示一些額外的關(guān)于錯誤的信息。下面是使用異常過濾器的示例。

應(yīng)用到Index動作方法上:

Asp.Net MVC過濾器的示例分析

在Views/Shared文件夾下添加一個顯示異常信息的視圖頁SpecialError.cshtml,頁面代碼如下:

@model HandleErrorInfo
  <!DOCTYPE html>
  <html>
  <head>
    <meta name="viewport" content="width=device-width" />
    <title>SpecialError</title>
  </head>
  <body>
    <p>
      <p>
        There was a<b>@Model.Exception.GetType().Name</b>
        while rendering<b>@Model.ControllerName</b>'s
        <b>@Model.ActionName</b> action
      </p>
    </p>
  </body>
</html>

運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

創(chuàng)建自定義的異常過濾器

如果我們對異常過濾器有特殊的需求,可以通過自定義的異常過濾器來完成,創(chuàng)建自定義異常過濾器必須實(shí)現(xiàn)IExceptionFilter接口,該接口代碼如下:

Asp.Net MVC過濾器的示例分析

當(dāng)一個未知處理異常發(fā)生時,OnException方法會被調(diào)用。該方法的傳遞一個ExceptionContext對象,派生于ControllerContext類,定義了一些額外的過濾器專有屬性如下表所示:

Asp.Net MVC過濾器的示例分析

拋出的異常通過Exception屬性是可以訪問的。通過把ExceptionHandled屬性設(shè)置為true,一個異常過濾器可以報告它已經(jīng)處理了該異常,應(yīng)用于一個動作的所有異常過濾器都會被調(diào)用。

需要注意的是:如果一個動作方法的所有異常過濾器均為把ExceptionHandled屬性設(shè)置為true,MVC框架將使用默認(rèn)的ASP.NET異常處理程序。

Result屬性有異常過濾器使用,以告訴MVC框架要做什么,異常過濾器的兩個主要應(yīng)用是記錄該異常到日志,并把適當(dāng)?shù)南@示給用戶。下面的代碼將演示通過創(chuàng)建一個自定義的異常過濾器,當(dāng)一個特定的鐘類的未處理異常出現(xiàn)時,把該用戶重定向到一個指定的錯誤頁面。

public class MyExectionAttribute:FilterAttribute,IExceptionFilter
  {
    public void OnException(ExceptionContext filterContext)
    {
      if(!filterContext.ExceptionHandled&&
        filterContext.Exception is NullReferenceException)
      {
        filterContext.Result = new RedirectResult("~/Content/SpecialErrorPage.html");
        filterContext.ExceptionHandled = true;
      }
    }
}

然后在項(xiàng)目根目錄添加一個名為Content的文件夾,在該文件夾下創(chuàng)建SpeciErrorPage.html文件,當(dāng)異常被處理時,將以這個錯誤頁面顯示個用戶。該頁面代碼如下:

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>
  <title></title>
</head>
<body>
  <h2>Sorry</h2>
  <p>this is a Excetption test</p>
  There was aNullReferenceException while renderingHome's Index action 
</body>
</html>

在控制器中應(yīng)用MyExection異常過濾器,并主動讓其拋出一個空引用異常,以便測試。

public class HomeController : Controller
  {
    [MyExection]
    public ActionResult Index()
    {
      throw new NullReferenceException();
    }
  }

運(yùn)行結(jié)果如下:

Asp.Net MVC過濾器的示例分析

以上是“Asp.Net MVC過濾器的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI