3

我正在将我的 .Net Core 2.2 应用程序迁移到 3.1 版。

我有以下动作过滤器:

public class MyFilterAttribute : ActionFilterAttribute
{
    private readonly string _name;
    private readonly int _num;
    private readonly string _type;

    public MyFilterAttribute(string name, int num, string type)
    {
        _name = name;
        _num = num;
        _type = type;
    }

    public override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
    {
        ...
        await base.OnActionExecutionAsync(context, next);
    }
}

我在 Startup.cs 文件中注册这个过滤器,如下所示:

services.AddTransient<MyFilterAttribute>();

我在我的控制器中使用过滤器,如下所示:

public class MyController : Controller
{
    [MyFilter("some-name", 5000, "some-type")]
    public async Task MyAction()
    {
        ...
    }
}

这个过滤器在我的 .Net Core 2.2 应用程序中运行良好。但是,现在当我尝试迁移到 3.1 版时,出现以下异常:

发生异常:CLR/System.AggregateException Microsoft.Extensions.DependencyInjection.dll 中发生“System.AggregateException”类型的未处理异常:“无法构造某些服务”找到内部异常,请参阅变量窗口中的 $exception更多细节。最里面的异常
System.InvalidOperationException:尝试激活“my_app.Filters.MyFilterAttribute”时无法解析“System.String”类型的服务。在 Microsoft.Extensions.DependencyInjection.ServiceLookup.CallSiteFactory.CreateArgumentCallSites(类型 serviceType,类型 implementationType,CallSiteChain callSiteChain,ParameterInfo [] 参数,布尔 throwIfCallSiteNotFound)在 Microsoft.Extensions.DependencyInjection.ServiceLookup.CallSiteFactory.CreateConstructorCallSite(ResultCache 生命周期,类型 serviceType,类型implementationType, CallSiteChain callSiteChain) 在 Microsoft.Extensions.DependencyInjection.ServiceLookup.CallSiteFactory.TryCreateExact(ServiceDescriptor 描述符, 类型 serviceType, CallSiteChain callSiteChain, Int32 slot) 在 Microsoft.Extensions.DependencyInjection。

我尝试像这样全局添加过滤器:

services.AddControllersWithViews(config =>
{
    config.Filters.Add<MyFilterAttribute>();
})

但我有同样的错误。

.Net Core 在 .Net Core 3.1 中读取自定义过滤器的方式是否发生了变化?

我在迁移重大更改文档中没有看到任何相关内容。

4

2 回答 2

1

根据这个(https://docs.microsoft.com/en-us/aspnet/core/mvc/controllers/filters?view=aspnetcore-3.1),您的所有操作过滤器都由 DI 解析,这意味着您的 2 个字符串参数将也得到解决。这导致了你的错误。

您可能需要考虑使用不需要由 DI 解析的不同类型的过滤器。

查看 2 种类型的过滤器

public class AddHeaderAttribute : ResultFilterAttribute
public class MyActionFilterAttribute : ActionFilterAttribute
于 2020-03-06T16:55:32.657 回答
0

Alex 和 Nkosi 的答案 (/comments) 是正确的,第一个实现的问题是过滤器是在依赖注入机制中构建的,而 DI 不知道如何处理字符串参数。

老实说,我不知道这是如何工作的,但我确实有一个 .Net Core 2.2 应用程序正在生产中,并以这种方式实现了过滤器。

无论如何,我将过滤器更改为 a TypeFilterAttribute,现在它在 .Net Core 3.1 中再次工作(无需在 DI 中注册):

public class MyFilterAttribute : TypeFilterAttribute
{
    public MyFilterAttribute(params object[] arguments) : base(typeof(MyFilterAttributeImpl))
    {
        Arguments = new object[] { arguments };
    }

    private class MyFilterAttributeImpl : ActionFilterAttribute
    {
        private readonly string _name;
        private readonly int _num;
        private readonly string _type;

        public MyFilterAttribute(object[] arguments)
        {
            _name = (string)arguments[0];
            _num = (int)arguments[1];
            _type = (string)arguments[2];
        }

        public override async Task OnActionExecutionAsync(ActionExecutingContext context, ActionExecutionDelegate next)
        {
            ...
            await base.OnActionExecutionAsync(context, next);
        }
    }
}

通过这个我们首先定义 aTypeFilterAttribute并在其中调用 an 的ActionFilterAttribute实现,我们可以使用漂亮的注解来调用过滤器:

[MyFilter("some-name", 5000, "some-type")]

代替:

[TypeFilter(typeof(MyFilterAttribute), Arguments = new object[] { "some-name", 5000, "some-type" })]

于 2020-03-07T22:31:31.387 回答