網站首頁 編程語言 正文
一、操作過濾器
1、定義
操作過濾器用于實現IActionFilter接口以及包裝操作方法執行。IActionFilter接口聲明兩個方法:OnActionExecuting和OnActionExecuted。OnActionExecuting在操作方法之前運行。OnActionExecuted在操作方法之后運行,可以執行其他處理,如向操作方法提供額外數據、檢查返回值或取消執行操作方法。
查看ActionFilterAttribute類的定義:
#region 程序集 System.Web.Mvc, Version=5.2.7.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35 // D:\Practice\MVC\自定義操作過濾器\MVCCustomerActionFilterDemo\packages\Microsoft.AspNet.Mvc.5.2.7\lib\net45\System.Web.Mvc.dll #endregion namespace System.Web.Mvc { // // 摘要: // 表示篩選器特性的基類。 [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, Inherited = true, AllowMultiple = false)] public abstract class ActionFilterAttribute : FilterAttribute, IActionFilter, IResultFilter { // // 摘要: // 初始化 System.Web.Mvc.ActionFilterAttribute 類的新實例。 protected ActionFilterAttribute(); // // 摘要: // 在執行操作方法后由 ASP.NET MVC 框架調用。 // // 參數: // filterContext: // 篩選器上下文。 public virtual void OnActionExecuted(ActionExecutedContext filterContext); // // 摘要: // 在執行操作方法之前由 ASP.NET MVC 框架調用。 // // 參數: // filterContext: // 篩選器上下文。 public virtual void OnActionExecuting(ActionExecutingContext filterContext); // // 摘要: // 在執行操作結果后由 ASP.NET MVC 框架調用。 // // 參數: // filterContext: // 篩選器上下文。 public virtual void OnResultExecuted(ResultExecutedContext filterContext); // // 摘要: // 在執行操作結果之前由 ASP.NET MVC 框架調用。 // // 參數: // filterContext: // 篩選器上下文。 public virtual void OnResultExecuting(ResultExecutingContext filterContext); } }
根據方法的名字就知道4個方法執行的順序了:
OnActionExecuting是Action執行前的操作、OnActionExecuted則是Action執行后的操作、OnResultExecuting是解析ActionResult前執行、OnResultExecuted是解析ActionResult后執行。
即:Action執行前:OnActionExecuting方法先執行→Action執行 →OnActionExecuted方法執行→OnResultExecuting方法執行→返回的ActionRsult中的 executeResult方法執行→OnResultExecuted執行。
2、案例
2.1、創建自定義操作過濾器
新建一個自定義過濾器,然后重新里面的方法,代碼如下:
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; namespace MVCCustomerActionFilterDemo.Extension { public class CustomerActionFilter :ActionFilterAttribute { public override void OnActionExecuting(ActionExecutingContext filterContext) { filterContext.HttpContext.Response.Write("Action方法準備執行"); base.OnActionExecuting(filterContext); } public override void OnActionExecuted(ActionExecutedContext filterContext) { filterContext.HttpContext.Response.Write("Action方法執行結束"); base.OnActionExecuted(filterContext); } } }
2.2、新建控制器
創建一個控制器,用來測試自定義操作過濾器,代碼如下:
using MVCCustomerActionFilterDemo.Extension; using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; namespace MVCCustomerActionFilterDemo.Controllers { public class ActionFiltersController : Controller { // GET: ActionFilters [CustomerActionFilter] public ActionResult Index() { Response.Write("執行Index...
"); return View(); } } }
Index方法對應的視圖代碼如下:
@{ Layout = null; }Index 操作過濾器測試頁面
運行結果;
二、結果過濾器
1、定義
結果篩選器用于實現IResultFilter接口以及包裝ActionResult對象的執行。IResultFilter接口聲明兩個方法OnResultExecuting和OnResultExecuted。OnResultExecuting在執行ActionResult對象之前運行。OnResultExecuted在結果之后運行,可以對結果執行其他處理,如修改 HTTP 響應。
結果過濾器也是實現了ActionFilterAttribute類。
2、案例
修改CustomerActionFilter類,重寫OnResultExecuting和OnResultExecuted,修改后的代碼如下:
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; namespace MVCCustomerActionFilterDemo.Extension { public class CustomerActionFilter :ActionFilterAttribute { public override void OnActionExecuting(ActionExecutingContext filterContext) { filterContext.HttpContext.Response.Write("Action方法準備執行"); base.OnActionExecuting(filterContext); } public override void OnActionExecuted(ActionExecutedContext filterContext) { filterContext.HttpContext.Response.Write("Action方法執行結束"); base.OnActionExecuted(filterContext); } public override void OnResultExecuting(ResultExecutingContext filterContext) { filterContext.HttpContext.Response.Write("Action方法執行結束,準備呈現視圖"); base.OnResultExecuting(filterContext); } public override void OnResultExecuted(ResultExecutedContext filterContext) { filterContext.HttpContext.Response.Write("視圖呈現結束"); base.OnResultExecuted(filterContext); } } }
運行結果:
三、案例
1、記錄操作
在真實項目中,可以利用操作過濾器記錄哪個用戶登錄系統以后進行了哪些操作。
1.1、創建實體類
新建用于記錄信息的實體類。代碼如下:
using System; using System.Collections.Generic; using System.Linq; using System.Web; namespace MVCCustomerActionFilterDemo.Models { public class LogEntity { ////// 控制器名稱 /// public string ControllerName { get; set; } ////// Action方法名稱 /// public string ActionName { get; set; } ////// 操作用戶id /// public string OperationUserId { get; set; } ////// 操作時間 /// public DateTime OperationTime { get; set; } } }
1.2、創建日志類
創建日志幫助類,代碼如下:
using MVCCustomerActionFilterDemo.Models; using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.IO; namespace MVCCustomerActionFilterDemo.Util { public class LogHelper { ////// 記錄操作日志 /// 這里為了方便測試記錄到txt文件里面,實際中應該是記錄到數據庫中 /// 然后有界面可以顯示這些操作記錄 /// /// public static void WriteOperRecore(LogEntity entity) { string strPath = @"C:\log.txt"; using (StreamWriter sw = new StreamWriter(strPath, true)) { sw.WriteLine("**************************"); sw.WriteLine($"操作時間:{entity.OperationTime}"); sw.WriteLine($"當前Controller名稱:{entity.ControllerName}"); sw.WriteLine($"當前Action名稱:{entity.ActionName}"); sw.WriteLine($"當前操作用戶id:{entity.OperationUserId}"); sw.Close(); } } } }
1.3、修改操作過濾器類
修改后的操作過濾器類代碼如下:
using MVCCustomerActionFilterDemo.Models; using MVCCustomerActionFilterDemo.Util; using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; namespace MVCCustomerActionFilterDemo.Extension { public class CustomerActionFilter : ActionFilterAttribute { public override void OnActionExecuting(ActionExecutingContext filterContext) { filterContext.HttpContext.Response.Write("Action方法準備執行"); string strControllerName = filterContext.RouteData.Values["controller"].ToString(); string strActionName = filterContext.RouteData.Values["action"].ToString(); LogEntity entity = new LogEntity() { OperationTime = DateTime.Now, ControllerName = strControllerName, ActionName = strActionName, // 為了方便測試寫admin,真實案例需要獲取當前登錄的用戶 OperationUserId = "admin" }; // 記錄操作記錄 LogHelper.WriteOperRecore(entity); base.OnActionExecuting(filterContext); } public override void OnActionExecuted(ActionExecutedContext filterContext) { filterContext.HttpContext.Response.Write("Action方法執行結束"); base.OnActionExecuted(filterContext); } public override void OnResultExecuting(ResultExecutingContext filterContext) { filterContext.HttpContext.Response.Write("Action方法執行結束,準備呈現視圖"); base.OnResultExecuting(filterContext); } public override void OnResultExecuted(ResultExecutedContext filterContext) { filterContext.HttpContext.Response.Write("視圖呈現結束"); base.OnResultExecuted(filterContext); } } }
運行程序,查看生成的日志:
2、實現權限控制功能
可以重寫OnActionExecuting方法實現授權過濾器一樣的功能,因為OnActionExecuting方法是在Action方法執行前執行的,自定義一個實現ActionFilterAttribute類的CustomerActionPremisFilters類,代碼如下:
using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; using MVCCustomerActionFilterDemo.DataBase; using MVCCustomerActionFilterDemo.Models; namespace MVCCustomerActionFilterDemo.Extension { public class CustomerActionPremisFilters :ActionFilterAttribute { public string ActionName { get; set; } //用于保存Action配置的別名 public string AreaName { get; set; } public string Roles { get; set; } public override void OnActionExecuting(ActionExecutingContext filterContext) { // 如果未登錄,則跳轉到登錄界面 if (!filterContext.HttpContext.User.Identity.IsAuthenticated) { filterContext.HttpContext.Response.Redirect("/Account/LogOn"); return; } //當前登錄用戶的用戶名 string userName = filterContext.HttpContext.User.Identity.Name; //當前登錄用戶對象 User user = SampleData.users.Find(u => u.UserName == userName); if (user != null) { //當前登錄用戶的角色 Role role = SampleData.roles.Find(r => r.Id == user.RoleId); //獲得controller: string controllerName = filterContext.RouteData.Values["controller"].ToString().ToLower(); if (ActionName == null) { ActionName = filterContext.RouteData.Values["action"].ToString(); } //查詢角色id RoleWithControllerAction roleWithControllerAction = SampleData.roleWithControllerAndAction.Find(r => r.ControllerName.ToLower() == controllerName && ActionName.ToLower() == ActionName.ToLower()); if (roleWithControllerAction != null) { //有權限操作當前控制器和Action的角色id this.Roles = roleWithControllerAction.RoleIds; } if (!string.IsNullOrEmpty(Roles)) { foreach (string roleid in Roles.Split(',')) { if (role.Id.ToString() == roleid) { //return就說明有權限了,后面的代碼就不跑了,直接返回視圖給瀏覽器就好 return; } } } filterContext.Result = new ViewResult { ViewName = "Error", }; return; } else { filterContext.Result = new EmptyResult(); filterContext.HttpContext.Response.Redirect("/Account/Logon", true); return; } } } }
新建ActionPremisFilters控制器,代碼如下:
using MVCCustomerActionFilterDemo.Extension; using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.Mvc; namespace MVCCustomerActionFilterDemo.Controllers { public class ActionPremisFiltersController : Controller { // GET: ActionPremisFilters [CustomerActionPremisFilters] public ActionResult Index() { return View(); } } }
修改SampleData數據,使角色id為2、3的可以訪問ActionPremisFilters的Index方法:
using MVCCustomerActionFilterDemo.Models; using System; using System.Collections.Generic; using System.Linq; using System.Web; namespace MVCCustomerActionFilterDemo.DataBase { ////// 測試數據(實際項目中,這些數據應該從數據庫拿) /// public class SampleData { public static Listusers; public static List roles; public static List roleWithControllerAndAction; static SampleData() { // 初始化用戶 users = new List () { new User(){ Id=1, UserName="jxl", RoleId=1}, new User(){ Id=2, UserName ="senior1", RoleId=2}, new User(){ Id=3, UserName ="senior2", RoleId=2}, new User(){ Id=5, UserName="junior1", RoleId=3}, new User(){ Id=6, UserName="junior2", RoleId=3}, new User(){ Id=6, UserName="junior3", RoleId=3} }; // 初始化角色 roles = new List () { new Role() { Id=1, RoleName="管理員", Description="管理員角色"}, new Role() { Id=2, RoleName="高級會員", Description="高級會員角色"}, new Role() { Id=3, RoleName="初級會員", Description="初級會員角色"} }; // 初始化角色控制器和Action對應類 roleWithControllerAndAction = new List () { new RoleWithControllerAction(){ Id=1, ControllerName="AuthFilters", ActionName="AdminUser", RoleIds="1"}, new RoleWithControllerAction(){ Id=2, ControllerName="AuthFilters", ActionName="SeniorUser",RoleIds="1,2"}, new RoleWithControllerAction(){ Id=3, ControllerName="AuthFilters", ActionName="JuniorUser",RoleIds="1,2,3"}, new RoleWithControllerAction(){ Id=3, ControllerName="AuthFilters", ActionName="Welcome",RoleIds="1,2"}, new RoleWithControllerAction(){ Id=4, ControllerName="ActionFilters", ActionName="Index", RoleIds="2,3"}, // 角色2、3可以訪問ActionPremisFilters控制器的Index方法 new RoleWithControllerAction(){ Id=4, ControllerName="ActionPremisFilters", ActionName="Index", RoleIds="2,3"} }; } } }
修改配置文件
測試,訪問ActionPremisFilters的Index方法,由于系統還沒有登錄,所以會跳轉到登錄頁面,這時候用jxl用戶登錄:
由于jxl用戶沒有訪問ActionPremisFilters控制器中Index方法的權限,所以會跳轉到Error頁面:
這時在用senior1用戶登錄,由于senior1用戶有權限訪問,所以會顯示Index視圖內容:
GitHub代碼地址:https://github.com/jxl1024/MVCCustomerActionFilterDemo
原文鏈接:https://www.cnblogs.com/dotnet261010/p/10845650.html
相關推薦
- 2023-01-15 Go語言使用組合的思想實現繼承_Golang
- 2023-06-16 GO的鎖和原子操作的示例詳解_Golang
- 2022-04-23 C語言復雜鏈表的復制實例詳解_C 語言
- 2022-05-22 Nginx安裝后常用功能配置基礎篇_nginx
- 2022-08-01 MongoDB創建與刪除數據庫_MongoDB
- 2021-12-03 Centos8環境下修改ssh端口號方法_Linux
- 2021-12-07 c++代碼各種注釋示例詳解_C 語言
- 2022-07-04 python設計模式之裝飾器模式_python
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支