AutoMapper的ProjectTo<>()无法找到映射

7
我有一个ASP.NET 5应用程序(运行在4.6.2上,不是Core)。我想使用AutoMapper的ProjectTo<>()方法将数据库中的结果投影到我的视图模型中。
我尝试了很多测试,但似乎仅使用ProjectTo<>()时无法找到映射。在不同位置使用相同的模型和视图模型进行mapper.Map<>()完美地工作。
我猜AutoMapper与我的DI(Autofac)的工作方式有问题,但我无法弄清楚原因。
无论如何,以下是代码:

Startup.Cs

 public IServiceProvider ConfigureServices(IServiceCollection services)
        {
            (...)

            // Autofac DI
            AutofacContainer = AutofacLoader.Configure(services).Build();

            return AutofacContainer.Resolve<IServiceProvider>();
        }

AutofacLoader.cs

public static ContainerBuilder Configure(IServiceCollection services)
        {
            var builder = new ContainerBuilder();

(...)


            // AutoMapper
            builder.RegisterModule<AutoMapperModule>();

            if (services != null)
            { 
                builder.Populate(services);

            }
            return builder;
        }

AutoMapperModule.cs

public class AutoMapperModule : Module
{
    protected override void Load(ContainerBuilder builder)
    {
        var mapping = new MapperConfiguration(cfg =>
        {
            cfg.AddProfile(new Core.Mappings.AutoMapperProfileConfiguration());
            cfg.AddProfile(new Dieet.Core.Mappings.AutoMapperProfileConfiguration());
        });
        builder.RegisterInstance(mapping.CreateMapper()).As<IMapper>().AutoActivate();
    }
}

测试失败,提示“从Patient到PatientViewModel的映射丢失,请使用Mapper.CreateMap创建”。
   [Fact]
    public async void InfohosServiceReturnsPatientViewModels()
    {
        var db = _container.Resolve<IInfohosDb>();

        var search = new PaginatedSearchBase();
        search.OrderBy = "Naam";

        var mapper = _container.Resolve<IMapper>();

        var result = await search.PagedResultAsAsync<Patient,PatientViewModel >(null,db.Patienten,mapper);
    }

PaginatedSearchBase

public class PaginatedSearchBase
{
    public string OrderBy { get; set; }
    public bool OrderDescending { get; set; }
    public int Page { get; set; } = 1;
    public int PageSize { get; set; } = 10;
}

最后是调用ProjectTo的扩展程序。
public static class PagedResultExtensions
{
    public static async Task<PagedResult<T>> PagedResultAsync<T>(this PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context) where T : class
    {
        int totalCount;
        var query = PrepareQuery(vm, whereCollection, context, out totalCount);

        return new PagedResult<T>
        {
            Results = await query.ToListAsync(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };
    }
    public static async Task<PagedResult<TAs>> PagedResultAsAsync<T, TAs>(this PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context, IMapper mapper) where T : class
    {
        int totalCount;
        var query = PrepareQuery(vm, whereCollection, context, out totalCount);

        return new PagedResult<TAs>
        {
----------> Results = await query.ProjectTo<TAs>(mapper).ToListAsync(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };
    }

    private static IQueryable<T> PrepareQuery<T>(PaginatedSearchBase vm, ICollection<Expression<Func<T, bool>>> whereCollection, IEnumerable<T> context,
        out int totalCount) where T : class
    {
        var query = context.AsQueryable();
        if (whereCollection != null)
        {
            foreach (var w in whereCollection)
            {
                if (w != null)
                {
                    query = query.Where(w);
                }
            }
        }
        // Order by
        query = query.OrderBy($"{vm.OrderBy} {(vm.OrderDescending ? "DESC" : "ASC")}");

        // Total rows
        totalCount = query.Count();

        // Paging
        query = query.Skip((vm.Page - 1)*vm.PageSize).Take(vm.PageSize);
        return query;
    }
}

提供信息,我正在使用以下版本:

  • "Autofac": "4.0.0-rc1-177"
  • "Autofac.Extensions.DependencyInjection": "4.0.0-rc1-177"
  • "AutoMapper": "4.2.1"

编辑:

我进行了一个新的测试来检查映射是否真正起作用:

var mapper = _container.Resolve<IMapper>();
        var p = new Patient();
        p.Naam = "Test";
        var vm = mapper.Map<PatientViewModel>(p);

        vm.Naam.ShouldBeEquivalentTo("Test");

This test passes
编辑2:
当我在Select()中使用Map<>时,它也可以工作,所以真正失败的是ProjectTo<>():
var results = await query.ToListAsync();
        return new PagedResult<TAs>
        {
            Results = results.Select(mapper.Map<TAs>).ToList(),
            Page = vm.Page,
            PageSize = vm.PageSize,
            Total = totalCount
        };

这个方案可行,但需要将映射器包含并非注入,并且不使用Automapper的ProjectTo进行数据库访问...

你可以尝试添加 "using AutoMapper.QueryableExtensions;" - Sam Zakhezin
我正在使用Visual Studio进行编码,如果不在“usings”中,编译器会产生错误。该方法被调用,但无法获取IMapper实例或其他某些东西,无法找到映射。 - Appsum Solutions
2个回答

13

我遇到了同样的问题,但是设法解决了。这是由于Automapper最近将整个API从使用静态方法转换为基于实例的方式,导致静态扩展方法不再知道映射配置,现在必须将它们传递给方法。最终我注册了一个MapperConfiguration实例作为IConfigurationProvider(我使用Unity)。

container.RegisterInstance(typeof (IConfigurationProvider), config);

这被注入到我的查询处理程序中:

[Dependency]
public IConfigurationProvider MapperConfigurationProvider { get; set; }

最后,MapperConfigurationProvider被传递给ProjectTo的调用:

.ProjectTo<Payment>(MapperConfigurationProvider);

希望这可以帮到你。


7
谢谢,添加MapperConfigurationProvider确实解决了问题!因为我的PagedResultAsAsync是静态的,所以我在方法中注入了IMapper。我更喜欢注入IMapper,因为如果需要,我可以使用mapper.Map()。我将语句改为:query.ProjectTo<TAs>(mapper.ConfigurationProvider).ToListAsync()。 - Appsum Solutions

5
您不需要专门将ConfigurationProvider添加到DI中。如果您已经将IMapper添加到DI中,则可以从Mapper本身读取ConfigurationProvider。例如:我曾经遇到过同样的问题,并创建了一个包含被注入的IMapper的基类:

public abstract class ServiceBase { public IMapper Mapper { get; set; } }

所有使用AutoMapper的服务都继承了这个类。现在,每当我的任何服务需要映射某些内容时,它们就会这样做:

    return context.SomeEntity
        .Where(e => e.Id == filter.Id)
        .ProjectTo<EntityDto>(Mapper.ConfigurationProvider).ToList();

注入 Mapper 后,只需将完全配置好的 Mapper 放入 DI 中即可。

container.Register(Component.For<IMapper>().UsingFactoryMethod(x =>
    {
        return new AutoMapperConfig().ConfigureMapper();
    })
    );

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接