zoukankan      html  css  js  c++  java
  • ASP.NET Core如何在ActionFilterAttribute里做依赖注入

    在ASP.NET Core里,我们可以使用构造函数注入很方便地对Controller,ViewComponent等部件做依赖注入。但是如何给过滤器ActionFilterAttribute也用上构造函数注入呢?

    640?wx_fmt=gif

    问题

    我的博客系统里有个用来删除订阅文件缓存的ActionFilter,想要在发生异常的时候记录日志。我的博客用的日志组件是NLog,因此不使用依赖注入的话,就直接使用LogManager.GetCurrentClassLogger()获得一个Logger的实例。整个过滤器的代码如下:

    public class DeleteSubscriptionCache : ActionFilterAttribute

    {

        private static readonly Logger Logger = LogManager.GetCurrentClassLogger();

        public override void OnActionExecuted(ActionExecutedContext context)

        {

            base.OnActionExecuted(context);

            DeleteSubscriptionFiles();

        }

        private void DeleteSubscriptionFiles()

        {

            try

            {

                // ...

            }

            catch (Exception e)

            {

                Logger.Error(e, "Error Delete Subscription Files");

            }

        }

    }

    然后在Action上去使用,和经典的ASP.NET MVC一样

    [Authorize]

    [HttpPost, ValidateAntiForgeryToken, DeleteSubscriptionCache]

    [Route("manage/edit")]

    public IActionResult Edit(PostEditModel model)

    这当然可以没有问题的运行,但写代码最重要的就是逼格,这个代码耦合了NLog,而我的博客系统里其他地方早就在用ASP.NET Core的ILogger接口了。如果哪天日志组件不再用NLog了,那么这个地方的代码就得改,而使用ILogger接口的代码就不需要动。虽然这种情况是绝对不会发生的,但是写代码一定要有追求,尽可能过度设计,才能不被人鄙视,然后才能面试造航母,工作拧螺丝。因此我决定把日志组件用依赖注入的方式安排一下。

    640?wx_fmt=gif

    改造过滤器

    方法和在Controller中使用依赖注入完全一样,我们使用构造函数注入ILogger<DeleteSubscriptionCache>类型。于是代码变成了这样:

    public class DeleteSubscriptionCache : ActionFilterAttribute

    {

        protected readonly ILogger<DeleteSubscriptionCache> Logger;

        public DeleteSubscriptionCache(ILogger<DeleteSubscriptionCache> logger)

        {

            Logger = logger;

        }

        public override void OnActionExecuted(ActionExecutedContext context)

        {

            base.OnActionExecuted(context);

            DeleteSubscriptionFiles();

        }

        private void DeleteSubscriptionFiles()

        {

            try

            {

                // ...

            }

            catch (Exception e)

            {

                Logger.LogError(e, "Error Delete Subscription Files");

            }

        }

    }

    但是问题来了,这样的话我们是没法在Action上无脑使用了,因为构造函数要求传参。如果要自己new一个的话,装逼就失败了。我们来看看正确的解决方法~

    ServiceFilter

    其实ASP.NET Core里,我们可以使用ServiceFilter来完成这个需求。它也是一种Attribute,可以作用在Action上。位于Microsoft.AspNetCore.Mvc.Core程序集里,定义如下:

    // A filter that finds another filter in an System.IServiceProvider.

    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = true, Inherited = true)]

    public class ServiceFilterAttribute : Attribute, IFilterFactory, IFilterMetadata, IOrderedFilter

    {

            public ServiceFilterAttribute(Type type);

            public int Order { get; set; }

            public Type ServiceType { get; }

            public bool IsReusable { get; set; }

            public IFilterMetadata CreateInstance(IServiceProvider serviceProvider);

    }

    ServiceFilter允许我们解析一个已经添加到IoC容器里的服务,因此我们需要把DeleteSubscriptionCache注册一下:

    services.AddScoped<DeleteSubscriptionCache>();

    然后就能直接使用了:

    [Authorize]

    [HttpPost, ValidateAntiForgeryToken]

    [ServiceFilter(typeof(DeleteSubscriptionCache))]

    [Route("manage/edit")]

    public IActionResult Edit(PostEditModel model)

    运行时发现ILogger已经能被实例化了,完美!

    640?wx_fmt=png

    参考资料:

    https://stackoverflow.com/questions/36109052/inject-service-into-action-filter/36109690

    https://docs.microsoft.com/en-us/aspnet/core/mvc/controllers/filters?view=aspnetcore-2.2

     
  • 相关阅读:
    转载文章----.NET 框架浅析
    转载文章----IL反编译利器——Ildasm.exe和Reflector.exe:
    转载文章----初识Ildasm.exe——IL反编译的实用工具
    北航物理实验
    计算机组成原理往年试题以及答案(tzf!!!)
    Java 往年试卷参考答案!!!
    算法导论期末考试英文练习题带答案!!!
    算法导论( FFT & 自动机 & 最优二叉搜索树 !!!)
    Java部分总结图片版2(已加上原图链接!!!)
    Java部分总结图片版(已经加上原图链接下载!!!)
  • 原文地址:https://www.cnblogs.com/webenh/p/11605560.html
Copyright © 2011-2022 走看看