如何在.NET Core中使用AutoMapper编写自定义解析器

本文将介绍如果利用.NET Core的依赖注入技术使用AutoMapper的自动逸解析器。

AutoMapper 是 .NET 生态系统中的一个出色工具,它简化了将属性值从一个对象“ClassOne”“映射”到另一个对象(可能是“ClassTwo”)的属性的过程。对于大多数用例,映射值可以很容易地从源对象确定,并且通常映射到目标对象上的一个字段,即使与该字段的名称不同,或者需要忽略的属性值。.

前几天,我遇到了一个edge案例,需要在特定映射中使用应用程序配置中的URL值。如何访问我在应用程序启动时注册为依赖项的配置并不明显。幸运的是,我在AutoMapper偶然发现了IValueResolver,我想和大家分享以下我是怎么处理的,这样,如果你遇到这种情况,就可以节约你的时间了。

自定义解析器

如上所述,我需要在程序启动期间注册为可注入依赖项的配置中的一个值。我需要创建一个自定义解析器来解决这个问题,它实现了一个名为 `IValueResolver` 的 AutoMapper的 接口。此接口要求您实现一个名为 Resolve的方法,该方法提供对源对象和目标对象的访问,并将目标值设置为函数的输出。

在这个解析器中,我们可以声明一个构造函数,它允许我们注入我们在启动时注册的配置对象。如果没有这个解析器,我们就无法在常规的 MappingProfile 中注入该配置,也不会给我们的顶级映射类引入一些非常紧密的耦合。使用这种方法,我们只在特定映射中需要它们时才注入依赖项。

using AutoMapper;
using Microsoft.Extensions.Configuration

public class MyResolver : IValueResolver<SourceType, TargetType, string>
{
  private readonly string _myAppUrl
  
  public MyResolver(IConfiguration configuration)
  {
    _myAppUrl = configuration["MyAppUrl"];
  }
  
  public string Resolve(SourceType source, TargetType target, string destMember, ResolutionContext context)
  {
    return $"{_myAppUrl}/api/Orders/{source.OrderId}";
  }
}

使用自定义解析器

我们在下面的代码中添加属性映射,并告诉 AutoMapper 在映射值时使用自定义解析器。

using AutoMapper;
public class MappingProfile : Profile
{
  public MappingProfile()
  {
    CreateMap<SouceType, TargetType>()
      .ForMember(x => x.Url, opt => opt.MapFrom<MyResolver>());
  }
}
// In Startup.cs
public void ConfigureServices(IServiceCollection services)
{
  // May not need this if mappings are in same project
  services.AddTransient<MyResolver>();
}

总结

我希望你喜欢这篇文章。AutoMapper 和 .NET Core 的 DI 都是很棒的工具,而自定义解析器使我们能够利用两者的强大功能。快乐学习,开心工作!