使用程序集扫描注册 ASP.NET Core JsonConverter 实现

问题描述

我有很多来自 JsonConverter<T> 的 json 转换器。它们位于各种程序集中,而不仅仅是执行程序集。

我是这样注册的:

services
  .AddControllers()
  .AddJsonoptions(o => {
    o.JsonSerializerOptions.Converters.Add(new FoojsonConverter());
    o.JsonSerializerOptions.Converters.Add(new BarjsonConverter());
    o.JsonSerializerOptions.Converters.Add(new BazJsonConverter());
    o.JsonSerializerOptions.Converters.Add(new QuxJsonConverter());
    // lots more...
  });

我宁愿使用程序集扫描动态添加它们 - 就像 AutoMapper、FluentValidation、Autofac 等一样。我查看了他们的存储库以寻找秘制酱汁,但找不到。

所以我尝试了一些基本的反射:

services
  .AddControllers()
  .AddJsonoptions(o => {

    var jsonConverters =
      AppDomain.CurrentDomain
      .GetAssemblies()
      .SelectMany(x => x.ExportedTypes)
      .Where(x => x.BaseType != null && x.IsAssignableFrom(typeof(JsonConverter<>)));

    foreach (var jsonConverter in jsonConverters)
      o.JsonSerializerOptions.Converters.Add(Activator.CreateInstance(jsonConverter) as JsonConverter);

  });

但是启动失败,并抛出:

超话 | Microsoft.AspNetCore.Hosting.Diagnostics |应用程序启动异常
Autofac.Core.DependencyResolutionException:激活 Microsoft.AspNetCore.Mvc.Infrastructure.ControllerActionEndpointDataSourceFactory -> Microsoft.AspNetCore 时引发异常 .Mvc.Infrastructure.DefaultActionDescriptorCollectionProvider -> λ:Microsoft.AspNetCore.Mvc.Abstractions.IActionDescriptorProvider[] -> Microsoft.AspNetCore.Mvc.ApplicationModels.ControllerA ctionDescriptorProvider -> Microsoft.AspNetCore.Mvc.ApplicationModels.ApplicationModelFactory -> λ:Microsoft.AspNetCore.Mvc.ApplicationModels.IApplicationModelProvider[] -> Microsoft.AspNetC ore.Mvc.ApplicationModels.DefaultApplicationModelProvider -> Microsoft.AspNetCore.Mvc.ModelBinding.Metadata.DefaultModelMetadataProvider -> λ:Microsoft.AspNetCore.Mvc.ModelBinding.Metadata.I CompositeMetadataDetailsProvider.
---> System.MissingMethodException: 无法创建抽象类。
在 System.RuntimeTypeHandle.CreateInstance(RuntimeType type,Boolean publicOnly,Boolean wrapExceptions,Boolean& canBeCached,RuntimeMethodHandleInternal& ctor,Boolean& hasNoDefaultCtor )

注意 Cannot create an abstract class。基类是抽象的,但我正在尝试激活子类。

这样做的正确方法是什么?另外,引用 AppDomain 是否合适,因为我记得在某处读到过该行为在 aspnet 中发生了变化。

解决方法

只需更改为:

var jsonConverters =
      AppDomain.CurrentDomain
      .GetAssemblies()
      .SelectMany(x => x.ExportedTypes)
      .Where(x => x.BaseType != null && x.IsAssignableTo(typeof(JsonConverter)));
,

JsonConverter 满足您的条件 x => x.BaseType != null && x.IsAssignableFrom(typeof(JsonConverter<>)。 它具有基本类型并且是可分配的。对不起;)

你总是可以检查是否输入 .IsAbstract 然后忽略它们。 另一种解决方案是按约定注册类(在这种情况下按名称)。我使用 Autofac 但方法是相同的

  builder
                .RegisterAssemblyTypes(serviceAssembly)
                .Where(t => t.Name.EndsWith("Service"))
                .AsSelf();
,

这对我有用:

services.AddControllers().AddJsonOptions(o => {

  var jsonConverters =
    AppDomain.CurrentDomain
    .GetAssemblies()
    .Where(x => !x.IsDynamic)
    .SelectMany(x => x.ExportedTypes)
    .Where(x => x.BaseType != null)                           // is not Object,or an interface
    .Where(x => x.BaseType.IsGenericType)                     // must derive from JsonConverter<T>
    .Where(x => x.BaseType.BaseType == typeof(JsonConverter)) // exclude framework types

  foreach (var jsonConverter in jsonConverters) {
    var instance = Activator.CreateInstance(jsonConverter) as JsonConverter;
    o.JsonSerializerOptions.Converters.Add(instance);
  }
});

我不确定是使用 AppDomain.CurrentDomain 还是 AssemblyLoadContext.Default,但这种方式似乎有效。