经过一周的时间没有分享文章了,主要是在使用.netcore做一个小的项目,项目面向大众用户的增删改查都做的差不多了,打算本周在云服务器上部署试试,很期待,也希望上线后大家多多支持;以上纯属个人废话,来一起看看今天的正篇环节:
.继承IActionFilter来定义个验证登录的流程例子
.ActionFilter中怎么使用依赖注入
下面一步一个脚印的来分享:
.继承IActionFilter来定义个验证登录的流程例子
首先,咋们定义一个名叫FilterTestController的Controller,返回一个JsonResult结果;为了例子查看的方便性,同时在此文件中定义一个名MoResponse的类,对应的代码如下:
public class FilterTestController : Controller
{
public JsonResult Index()
{
var response = new MoResponse();
response.Status = ; return Json(response);
}
} public class MoResponse
{ public int Status { get; set; }
public string Des { get; set; } = "Ok";
}
然后,执行一下运行命令dotnet run,浏览器中输入默认端口好5000并访问刚才定义的Action,地址如:http://localhost:5000/FilterTes,不出意外大家都能看到如下结果的图样:
这表示项目初建没有问题;
接着,创建一个类,名称为CheckLoginAttribute并且继承和实现Attribute, IActionFilter,这里实现IActionFilter接口的OnActionExecuting,OnActionExecuted两个方法,看方法名称后缀大概就明白一个是Action执行之前,一个是Action方法执行之后调用的,这里简单补充下以前的mvc版本大部分都是去继承ActionFilterAttribute,其实这个也去继承和实现了Attribute, IActionFilter,不过还有其他的扩展罢了,这里不详细区分说明:
然后咋们分别在两个OnActionExecut中增加一些输入信息,并且在FilterTestController的Index方法上方增加[CheckLogin]标记,代码如下:
public class CheckLoginAttribute : Attribute, IActionFilter
{ public void OnActionExecuted(ActionExecutedContext context)
{ Console.WriteLine(DateTime.Now + "end...");
} public void OnActionExecuting(ActionExecutingContext context)
{ Console.WriteLine(DateTime.Now + "start..."); //context.Result = new RedirectResult("http://www.cnblogs.com/"); }
}
再运行起来,访问刚才的路由,可以在命令窗体看到如图:
刚才说的CheckLoginAttribute中的OnActionExecuting对应的是调用Controller的Action方法之前执行的部分,上面代码刚才注释的部分context.Result = new RedirectResult("http://www.cnblogs.com/");是跳转到该http://www.cnblogs.com/地址中去,如果放开注释,看到的效果是访问之前Controller路由后直接跳转到了该地址,并且名利窗体中只有start...日志的部分,没有了之前end...的部分,可以看出这里context.Result效果是直接终止了程序继续往下执行;到这里就可以走一个简单登录的验证了,我这里只说下流程,不做具体代码,因为这不是重点哈哈:
1.在OnActionExecuting方法中使用context.HttpContext.Session获取用户登陆的session(当然其他session保存方式除外)
2.使用context.HttpContext.Request.Path获取当前访问的路由地址
3.如果session为空,使用context.Result = new RedirectResult("/Login?returnUrl=" + context.HttpContext.Request.Path);跳转到路由Login中去,returnUrl参数使用来传递登陆有再跳转到当前访问地址中去
.ActionFilter中怎么使用依赖注入
这个是值得关注的地方,这个在我的项目最初写的时候遇到的问题;下面是一些分析,可供大家参考:
1.netcore常用的注入方式是通过构造函数注入的
2.通过构造函数注入后,在需要使用Filter的Action中无法通过对应参数个数的构造函数调用;只能调用无参的Filter构造函数
3.注意在需要使用依赖注入的Filter中不用定义无参构造函数(这里实验过了,如果定义TypeFilter将会以无参构造函数为优先创建实例,这样将会是依赖注入失败)
以上就是最开始无法直接使用结构器依赖注入的原因,后面无意中发现一个很有用的过滤器:TypeFilterAttribute,该过滤器可以通过构造函数传递进去的对象实例化,下面我们一起来看下:
首先,我们定义个MyActionFilterAttribute类并且继承TypeFilterAttribute,默认继承构造函数,然后在Controller的Index上方使用这个自定义属性并且传递我们定义的CheckLoginAttribute类型为参数,如下代码:
[MyActionFilter(typeof(CheckLoginAttribute))]
public JsonResult Index()
{
var response = new MoResponse();
response.Status = ; return Json(response);
}
MyActionFilterAttribute代码如下:
public class MyActionFilterAttribute : TypeFilterAttribute
{
public MyActionFilterAttribute(Type type) : base(type)
{ }
}
好了,咋们再自定义个简单的服务,并且把定义的服务在Startup.cs文件中增加代码services.AddTransient<LogService>();注入服务,服务要求是定义个方法,在命令框中输出hello...,如下代码:
public class LogService
{ public async void _LogRequest()
{ await Task.Run(() =>
{ for (int i = ; i < ; i++)
{
Console.WriteLine(DateTime.Now + "hello...");
}
});
}
}
再来,CheckLoginAttribute中增加代码如:
public class CheckLoginAttribute : Attribute, IActionFilter
{
private readonly LogService _logService;
public CheckLoginAttribute(LogService logService)
{ _logService = logService;
} public void OnActionExecuted(ActionExecutedContext context)
{ Console.WriteLine(DateTime.Now + "end...");
} public void OnActionExecuting(ActionExecutingContext context)
{ Console.WriteLine(DateTime.Now + "start..."); var path = context.HttpContext.Request.Path;
//context.Result = new RedirectResult($"/Login?returnUrl={path}"); _logService._LogRequest();
}
}
好了咋们一起dotnet run看到的效果如:
这个依赖注入到Filter中就成功了,其实上面定义的MyActionFilterAttribute也可以看做多余吧,因为在Action上使用的其实就是TypeFilterAttribute自身的构造函数方法,咋们可以直接在Action上使用TypeFilter如图:
两者效果是一样的,只是定义一个Filter可以记录其他的日志信息或干其他的事情罢了;这次分享的内容就是这样了,不知道描述的是否清晰,希望多多支持,谢谢。