我正在尝试使用 AutoMapper 通过多对多表将实体映射到集合中。
我的域模型( Entity Framework )如下所示:
public class User
{
public int UserID { get; set; }
public string Name { get; set; }
public IList<UserCompany> UserCompanies { get; set; }
}
public class Company
{
public int CompanyID { get; set; }
public string Name { get; set; }
}
public class UserCompany
{
public int UserCompanyID { get; set; }
public int UserID { get; set; }
public int CompanyID { get; set; }
}
我试图映射到一个看起来像这样的类:
public class CompanyDTO
{
public int CompanyID { get; set; }
public string Name { get; set; }
}
public class UserDTO
{
public int UserID { get; set; }
public string Name { get; set; }
public IList<CompanyDTO> Companies { get; set; }
}
我当前的映射配置如下所示:
Mapper.CreateMap<Company, CompanyDTO>(); //works no problem
Mapper.CreateMap<User, UserDTO>()
.ForMember( dto => dto.Companies,
opt => opt.MapFrom(x => Mapper.Map<IList<Company>, IList<CompanyDTO>>(x.UserCompanies.Select( y => y.Company ).ToList())));
断言配置有效返回 true,但是当我尝试将 User 实际映射到 UserDTO 时,我得到:方法实现中的主体签名和声明不匹配。
如果我使用 AfterMap 并手动将每个公司移到公司列表中,它将起作用,但似乎我应该能够在创建 map 中处理此问题。
在我从数据库获取单个用户的查询中,我包含了 UserCompany.Company 导航属性,我可以快速查看并查看正在返回的内容。
最佳答案
您不需要在映射配置中具有显式映射。您应该能够执行以下操作:
Mapper.CreateMap<User, UserDTO>()
.ForMember(dto => dto.Companies, opt => opt.MapFrom(x => x.UserCompanies));
您还需要为 UserCompany 定义一个映射:
Mapper.CreateMap<UserCompany, CompanyDTO>();
请注意,您的示例中没有 CompanyDTO 类,因此我无法确定实际的映射配置。
更新
我认为在您的用户实体中需要一个 IList 而不仅仅是一个 IList 是有原因的。鉴于此,我相信您需要一个自定义解析器:https://github.com/AutoMapper/AutoMapper/wiki/Custom-value-resolvers
更新 2
我很高兴你把它整理好了。为了完整起见,下面是您决定对上述类使用自定义 ValueResolver 的示例。
Mapper.CreateMap<Company, CompanyDTO>();
Mapper.CreateMap<User, UserDTO>()
.ForMember(dto => dto.Companies, opt => opt.ResolveUsing<CompanyResolver>());
Mapper.AssertConfigurationIsValid();
CompanyResolver 是这样的
public class CompanyResolver : ValueResolver<User, IList<CompanyDTO>>
{
protected override IList<CompanyDTO> ResolveCore(User source)
{
return source.UserCompanies
.Select(userCompany =>
Mapper.Map<Company, CompanyDTO>(companies.FirstOrDefault(x => x.CompanyID == userCompany.CompanyID)))
.ToList();
}
}
关于AutoMapper 多对多关系转化为集合,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11782948/