AutoMapper.ProjectTo 具有字段级类型转换
AutoMapper.ProjectTo with field level type conversion
我正在尝试使用 AutoMapper 的 ProjectTo 扩展方法将基于 Entity Framework table 的对象转换为自定义 DTO。 EF 端的 Person 实体如下所示:
public partial class Person
{
public int PersonId { get; set; }
public string Name { get; set; }
public string Age { get; set; }
}
我要投影到的 PersonViewModel DTO 如下所示:
public class PersonViewModel
{
public PersonViewModel(Person p)
{
PersonId = p.PersonId;
Name = p.Name;
Age = Convert.ToInt32(p.Age);
}
public PersonViewModel()
{
}
public PersonViewModel(int id, string name, int age)
{
PersonId = id;
Name = name;
Age = age;
}
public int PersonId { get; set; }
public string Name { get; set; }
public int Age { get; set; }
}
需要注意的关键是人 table 包含一个名为 Age 的字段,该字段是一个 nvarchar 类型,而 PersonViewModel 具有一个 int 类型的 Age。不幸的是,我无法更改 table,必须找到一种方法来转换数据。
以下代码是我尝试过的 AutoMapper 配置,它们的 "issues" 和消费代码(数据库调用等)。
MapperConfiguration config = new MapperConfiguration(cfg =>
{
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>();
// ----- Throws "Linq to Entities does not support method" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ForMember(dest => dest.Age, opt => opt.MapFrom(src => Convert.ToInt32(src.Age)));
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel(src.PersonId, src.Name, Convert.ToInt32(src.Age)));
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel(src.PersonId, src.Name, 25));
// ----- Throws "Linq to Entities does not support method" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ProjectUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructUsing(src => new PersonViewModel(src));
// ----- Doesn't throw exception, but all values are null or 0. ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConvertUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Doesn't throw exception, but all values are null or 0. ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConvertUsing(src => new PersonViewModel(src.PersonId, src.Name, Convert.ToInt32(src.Age)));
// ----- Works, but bypasses the conversion I need ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ProjectUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = 25
// });
// ----- Works, but bypasses the conversion I need ------
cfg.CreateMap<Person, PersonViewModel>()
.ForMember(dest => dest.Age, opt => opt.MapFrom(src => 35));
});
using (FamilyEntities db = new FamilyEntities())
{
PersonViewModel p = db.Set<Person>()
.Where(x => x.PersonId == 1)
.ProjectTo<PersonViewModel>(config)
.SingleOrDefault();
Console.WriteLine("Name: " + p.Name + Environment.NewLine + "Age: " + p.Age);
}
Console.ReadLine();
是否可以使用 AutoMapper 的 ProjectTo 将字符串转换为 int?其次,如果可以的话,那个配置是什么样的?
在此先感谢您的帮助。
你的第一次尝试是为什么这行不通。 Entity Framework 不知道如何将字符串转换为 int,所以您很吃力。 None 其他方式很重要,重要的是 EF 给你的例外。
如果 EF 不支持转换,AutoMapper 将无能为力。
相反,我会在您的视图模型中执行类似转换的操作:
public class PersonViewModel
{
public int PersonId { get; set; }
public string Name { get; set; }
public string Age { get; set; }
public int AgeValue => Convert.ToInt32(Age);
}
解释了如何向 EF 添加功能。基本上,生成的查询必须执行类型转换。
我正在尝试使用 AutoMapper 的 ProjectTo 扩展方法将基于 Entity Framework table 的对象转换为自定义 DTO。 EF 端的 Person 实体如下所示:
public partial class Person
{
public int PersonId { get; set; }
public string Name { get; set; }
public string Age { get; set; }
}
我要投影到的 PersonViewModel DTO 如下所示:
public class PersonViewModel
{
public PersonViewModel(Person p)
{
PersonId = p.PersonId;
Name = p.Name;
Age = Convert.ToInt32(p.Age);
}
public PersonViewModel()
{
}
public PersonViewModel(int id, string name, int age)
{
PersonId = id;
Name = name;
Age = age;
}
public int PersonId { get; set; }
public string Name { get; set; }
public int Age { get; set; }
}
需要注意的关键是人 table 包含一个名为 Age 的字段,该字段是一个 nvarchar 类型,而 PersonViewModel 具有一个 int 类型的 Age。不幸的是,我无法更改 table,必须找到一种方法来转换数据。
以下代码是我尝试过的 AutoMapper 配置,它们的 "issues" 和消费代码(数据库调用等)。
MapperConfiguration config = new MapperConfiguration(cfg =>
{
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>();
// ----- Throws "Linq to Entities does not support method" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ForMember(dest => dest.Age, opt => opt.MapFrom(src => Convert.ToInt32(src.Age)));
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel(src.PersonId, src.Name, Convert.ToInt32(src.Age)));
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructProjectionUsing(src => new PersonViewModel(src.PersonId, src.Name, 25));
// ----- Throws "Linq to Entities does not support method" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ProjectUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Throws "Unable to create map expression" exception ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConstructUsing(src => new PersonViewModel(src));
// ----- Doesn't throw exception, but all values are null or 0. ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConvertUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = Convert.ToInt32(src.Age)
// });
// ----- Doesn't throw exception, but all values are null or 0. ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ConvertUsing(src => new PersonViewModel(src.PersonId, src.Name, Convert.ToInt32(src.Age)));
// ----- Works, but bypasses the conversion I need ------
//cfg.CreateMap<Person, PersonViewModel>()
// .ProjectUsing(src => new PersonViewModel
// {
// PersonId = src.PersonId,
// Name = src.Name,
// Age = 25
// });
// ----- Works, but bypasses the conversion I need ------
cfg.CreateMap<Person, PersonViewModel>()
.ForMember(dest => dest.Age, opt => opt.MapFrom(src => 35));
});
using (FamilyEntities db = new FamilyEntities())
{
PersonViewModel p = db.Set<Person>()
.Where(x => x.PersonId == 1)
.ProjectTo<PersonViewModel>(config)
.SingleOrDefault();
Console.WriteLine("Name: " + p.Name + Environment.NewLine + "Age: " + p.Age);
}
Console.ReadLine();
是否可以使用 AutoMapper 的 ProjectTo 将字符串转换为 int?其次,如果可以的话,那个配置是什么样的?
在此先感谢您的帮助。
你的第一次尝试是为什么这行不通。 Entity Framework 不知道如何将字符串转换为 int,所以您很吃力。 None 其他方式很重要,重要的是 EF 给你的例外。
如果 EF 不支持转换,AutoMapper 将无能为力。
相反,我会在您的视图模型中执行类似转换的操作:
public class PersonViewModel
{
public int PersonId { get; set; }
public string Name { get; set; }
public string Age { get; set; }
public int AgeValue => Convert.ToInt32(Age);
}
解释了如何向 EF 添加功能。基本上,生成的查询必须执行类型转换。