前言
上两篇文章分享了过滤器实现JWT进行鉴权,分别是通过授权过滤器和操作过滤器实现,这两个过滤器也是最常用的。文章链接:授权过滤器—MVC中使用授权过滤器实现JWT权限认证,操作过滤器—MVC中使用操作过滤器实现JWT权限认证,接下来将简单的谈谈资源过滤器在MVC中如何使用,一般项目中这个过滤器很少用到。
一、什么是资源过滤器?
过滤器(Filter)是 AOP(面向切面编程) 思想的一种实现,供我们在执行管道的特定阶段执行代码,通过使用过滤器可以实现短路请求、缓存请求结果、日志统一记录、参数合法性验证、异常统一处理、返回值格式化 等等,同时使业务代码更加简洁单纯,避免很多重复代码。所以在我们的过滤器中,大部分过滤器有开始执行action,即ing 状态的方法,也有action业务代码执行完后触发的ed状态的方法。
资源过滤器在过滤器管道中第二个被执行,通常用于请求结果的缓存和短路过滤器管道,通过实现接口 IResourceFilter 或者IAsyncResourceFilter。和其他过滤器一样,实现接口,只是接口不同,接收的参数两类型不同,但是这也正意味着执行的时机不同。接收的参数类型为:ResourceExecutedContext
。
二、资源过滤器实现
资源过滤器定义:
资过滤器的定义,需要实现接口 IResourceFilter 或者IAsyncResourceFilter,接收的参数类型为:ResourceExecutedContext。
/// <summary>
/// 资源过滤器
/// </summary>
public class MyResourceFilter : Attribute, IResourceFilter//过滤器要继承Attribute 特性,这样我们也可以当做特性使用
{
/// <summary>
/// 内存缓存对象
/// </summary>
private readonly IMemoryCache _memoryCache;
/// <summary>
/// 构造注入
/// </summary>
public MyResourceFilter(IMemoryCache memoryCache)
{
_memoryCache = memoryCache;
}
/// <summary>
/// 资源过滤器过滤器执行之前(befor)
/// </summary>
/// <param name="context"></param>
public void OnResourceExecuted(ResourceExecutedContext context)
{
//针对哪些Action,也可以吧特性标注在Action上
var ad = context.ActionDescriptor;
var str = ad.RouteValues["controller"] + "/" + ad.RouteValues["action"];
if (str != "ResourceFilter/Test")
{
return;
}
//我们可以将当前的结果context.Result缓存起来,当执行ing时,直接返回,为了方便示例演示,我们用时间表示。
string content = "Action第一次执行调用时间:" + DateTime.Now;
var value = _memoryCache.Get("key"); //判断内存中是否有内容,有就直接返回,不再执行action过程。
if (value == null)
{
_memoryCache.Set("key", content);
}
}
/// <summary>
/// 资源过滤器过滤器执行之后(after)
/// </summary>
/// <param name="context"></param>
public void OnResourceExecuting(ResourceExecutingContext context)
{
//针对哪些Action,也可以吧特性标注在Action上
var ad = context.ActionDescriptor;
var str = ad.RouteValues["controller"] + "/" + ad.RouteValues["action"];
if (str != "ResourceFilter/Test")
{
return;
}
var content = _memoryCache.Get("key"); //判断内存中是否有内容,有就直接返回,不再执行action过程。
if (content != null)
{
var result = new { IsSuccess = true, Msg= _memoryCache.Get("key") };
//短路返回,不会再执行Action中方法和OnResourceExecuted方法
context.Result = new ContentResult() { Content = Newtonsoft.Json.JsonConvert.SerializeObject(result) };
};
}
}
添加到全局过滤器:
services.AddMvc(options =>
{
options.Filters.Add<MyResourceFilter>();
});
添加测试Action:
/// <summary>
/// 资源过滤器测试
/// </summary>
public class ResourceFilterController : ControllerBase
{
/// <summary>
/// 内存缓存对象
/// </summary>
private readonly IMemoryCache _memoryCache;
/// <summary>
/// 构造注入
/// </summary>
/// <param name="memoryCache"></param>
public ResourceFilterController(IMemoryCache memoryCache)
{
_memoryCache = memoryCache;
}
/// <summary>
/// 资源过滤器测试,获取接口第一次调用时间
/// 如果第一次调用Action:会进入Action执行方法体
/// 如果不是第一次调用Action:在资源过滤器中直接短路返回第一次调用时间
/// </summary>
/// <returns></returns>
[HttpGet]
public ActionResult Test()
{
string content = "Action第一次执行调用时间:" + DateTime.Now;
var value = _memoryCache.Get("key"); //判断内存中是否有内容,有就直接返回,不再执行action过程。
if (value == null)
{
_memoryCache.Set("key", content);
}
return Ok(new { IsSuccess = true, Msg = content });
}
}
三、验证:
第1次调用:
访问地址:
https://localhost:5001/ResourceFilter/Test
第N次调用:
建群声明:本着技术在于分享,方便大家交流学习的初心,特此建立【编程内功修炼交流群】,热烈欢迎各位爱交流学习的程序员进群,也希望进群的大佬能不吝分享自己遇到的技术问题和学习心得!