首页 > 解决方案 > 在 Entity Framework Core 中将字典转换为 ICollection

问题描述

在这里,我被EF Core中的dictionaryto转换困住了。IcollectionDictionaryFlatEmployee课堂上存储了数据库中的键、值对列表。我已经这样声明了:

public class FlatEmployee
{
    public int EmployeeId { get; set; }
    public Dictionary<string, long> PayAndAllowances { get; set; }
}

//====================Configuration 
public void Configure(EntityTypeBuilder<FlatEmployee> builder)
{
        builder.HasKey(f => new { f.EmployeeId });
        builder.Property(sb => sb.PayAndAllowances)
            .HasConversion(
                pa => JsonConvert.SerializeObject(pa),
                pa => JsonConvert.DeserializeObject<Dictionary<string, long>>(pa));
}

当我播种或插入时,这绝对可以正常工作。但是当我尝试获取 FlatEmployee 类时遇到的问题。这是因为我想在 Collection 中获取字典。为此,我声明了另一个这样的类:

public class LastPayCertificateViewModel: IHaveCustomMapping
{
    public int EmployeeCode { get; set; }
    public EmployeeEarningDTO PayAndAllowances { get; set; }

    public void CreateMappings(Profile configuration)
    {
        configuration.CreateMap<FlatEmployee, LastPayCertificateViewModel>()
            .ForMember(dto => dto.EmployeeCode , opt => opt.MapFrom(entity => entity.EmployeeId ));
    }
}

public class EmployeeEarningDTO : IHaveCustomMapping
{
    public ICollection<BaseEmployeeDictionary> PayAndAllowances { get; set; }

    public void CreateMappings(Profile configuration)
    {
        configuration.CreateMap<FlatEmployee, EmployeeEarningDTO>()
            .ForMember(dto => dto.PayAndAllowances, opt => opt.MapFrom(entity => entity.PayAndAllowances));
    }
}

public class BaseEmployeeDictionary
{
    public string Name { get; set; }
    public long Amount { get; set; }
}

当我尝试使用上述类来获取这样的数据时:

public class LastPayCertificateQuery : IRequest<LastPayCertificateViewModel>
{
    public string EmployeeCode { get; set; }
}

public async Task<LastPayCertificateViewModel> Handle(LastPayCertificateQuery request, CancellationToken cancellationToken)
{           
    var predicate = PredicateBuilder.False<FlatEmployee>();
    predicate = predicate.Or(emp => emp.EmployeeId == request.EmployeeCode);
    var employee = await _context.FlatEmployee
                        .Where(predicate)
                        .FirstOrDefaultAsync(cancellationToken);
    if (employee == null)
        return null;
    var empVM = _mapper.Map<LastPayCertificateViewModel>(employee);
}

然后我在PayAndAllowances in empVM. 这就是我的问题所在。我的问题在哪里?我认为这是因为 Dictionary 具有键值对并且无法转换为BaseEmployeeDictionary. 我也尝试过这种方式将列表项添加到PayAndAllowances in empVM

 foreach (KeyValuePair<string, long> data in employee.DeductionsByAdjustment)
 {
     BaseEmployeeDictionary listItem = new BaseEmployeeDictionary
     {
         Name = data.Key,
         Amount = data.Value
     };
     empVM.EarningDetails.PayAndAllowances.Add(listItem);
     return empVM;
  }

这当然不会起作用,因为它empVM.EarningDetails.PayAndAllowances是 null 并抛出 NullReferenceException。我的疑问是如何在创建地图时在字典和 ICollection 之间进行映射EmployeeEarningDTO。或者,如果您提出宝贵的建议和解决方案,我们将不胜感激。

标签: c#ef-code-firstentity-framework-coreasp.net-core-webapiasp.net-core-2.1

解决方案


原来是 AutoMapper 映射问题。

首先,与相比, EmployeeEarningDTOinside创建了额外的级别:LastPayCertificateViewModelFlatEmployee

LastPayCertificateViewModel.PayPayAndAllowances.PayAndAllowances

对比

FlatEmployee.PayAndAllowances

AutoMapper 默认映射同名的属性。所以在里面FlatEmployee映射LastPayCertificateViewModel它会尝试映射Dictionary<string, long> PayAndAllowancesEmployeeEarningDTO PayAndAllowances. 但是没有从Dictionary<string, long>到的映射EmployeeEarningDTO。相反,有一个从FlatEmployeeto的映射EmployeeEarningDTO,所以你必须告诉 AM 使用它:

configuration.CreateMap<FlatEmployee, LastPayCertificateViewModel>()
    .ForMember(dto => dto.EmployeeCode, opt => opt.MapFrom(entity => entity.EmployeeId))
    .ForMember(dto => dto.PayAndAllowances, opt => opt.MapFrom(entity => entity)); // <--

其次,映射 from FlatEmployeeto EmployeeEarningDTO- AM 会自动尝试映射PayAndAllowances属性,但是没有映射 from KeyValuePair<string, long>to BaseEmployeeDictionary。你可以定义这样的映射

configuration.CreateMap<KeyValuePair<string, long>, BaseEmployeeDictionary>()
                        .ForMember(dst => dst.Name, opt => opt.MapFrom(src => src.Key))
                        .ForMember(dst => dst.Amount, opt => opt.MapFrom(src => src.Value));

这将允许您简单地使用

configuration.CreateMap<FlatEmployee, EmployeeEarningDTO>();

但是您可能不会这样做,因为您不希望每个都 KeyValuePair<string, long>映射到BaseEmployeeDictionary,因此您可以就地进行映射:

configuration.CreateMap<FlatEmployee, EmployeeEarningDTO>()
     .ForMember(dto => dto.PayAndAllowances, opt => opt.MapFrom(entity => entity.PayAndAllowances
        .Select(src => new BaseEmployeeDictionary { Name = src.Key, Amount = src.Value })));

推荐阅读