[英]Dictionary <string,string> map to an object using Automapper
我有一个 class 之类的
public User class
{
public string Name{get;set;}
public string Age{get;set;
}
用像这样的字典
Dictionary<string,string> data= new Dictionary<string,string>();
data.Add("Name","Rusi");
data.Add("Age","23");
User user= new User();
现在我想使用Automapper
将 map User
object 添加到这本dictionary
。 Automapper 映射对象的属性,但在我的例子中有一个字典和 object。
这如何映射?
AutoMapper 在对象的属性之间映射,并且不应该在这种情况下运行。 在这种情况下,您需要反射魔法。 您可以通过中间序列化作弊:
var data = new Dictionary<string, string>();
data.Add("Name", "Rusi");
data.Add("Age", "23");
var serializer = new JavaScriptSerializer();
var user = serializer.Deserialize<User>(serializer.Serialize(data));
如果您坚持使用 AutoMapper,您可以例如执行以下操作:
Mapper
.CreateMap<Dictionary<string, string>, User>()
.ConvertUsing(x =>
{
var serializer = new JavaScriptSerializer();
return serializer.Deserialize<User>(serializer.Serialize(x));
});
接着:
var data = new Dictionary<string, string>();
data.Add("Name", "Rusi");
data.Add("Age", "23");
var user = Mapper.Map<Dictionary<string, string>, User>(data);
如果您需要处理带有子对象的更复杂的对象层次结构,您必须问自己以下问题:在这种情况下Dictionary<string, string>
是正确的数据结构吗?
这个线程有点旧,但现在有如何在没有任何配置的情况下在 automapper 上做到这一点,如官方文档所述:
AutoMapper 可以在没有任何显式配置的情况下映射到/从动态对象 (...) 同样,您可以直接从 Dictionary 映射到对象,AutoMapper 会将键与属性名称对齐。
更新:
以下代码显示了一个工作示例(带有单元测试)。
void Test()
{
var mapper = new MapperConfiguration(cfg => { }).CreateMapper();
var dictionary = new Dictionary<string, object>()
{
{ "Id", 1 },
{ "Description", "test" }
};
var product = mapper.Map<Product>(dictionary);
Assert.IsNotNull(product);
Assert.AreEqual(product.Id, 1);
Assert.AreEqual(product.Description, "test");
}
class Product
{
public int Id { get; set; }
public string Description { get; set; }
}
因为我刚刚偶然发现了这个问题,所以我想在当前版本的 AutoMapper 中添加这个答案(即使原始问题已经很老了):
public class MyConfig
{
public string Foo { get; set; }
public int Bar { get; set; }
}
var config = new MapperConfiguration(cfg => {});
var mapper = config.CreateMapper();
var source = new Dictionary<string, object>
{
["Foo"] = "Hello",
["Bar"] = 123
};
var obj = mapper.Map<MyConfig>(source);
obj.Foo == "Hello"; // true
AutoMapper 是一个非常灵活的解决方案。 您可能可以使用自定义映射配置文件来实现这一点,例如:
public class UserMappingProfile : Profile
{
// Props
public override string ProfileName { get { return "UserMappingProfile"; } }
// Methods
public override void Configure()
{
CreateMap<User, Dictionary<string, string>>().ConvertUsing<DictionaryTypeConverter>();
base.Configure();
}
// Types
internal class DictionaryTypeConverter : ITypeConverter<User, Dictionary<string, string>>
{
public User Convert(ResolutionContext context)
{
var dict = context.SourceValue as Dictionary<string, string>;
if (dict == null)
return null;
return new User() { Name = dict["Name"], Age = dict["Age"] };
}
}
}
有了这个,我可以创建一个映射器的自定义实例:
var config = new Configuration(new TypeMapFactory(), MapperRegistry.AllMappers());
config.AddProfile<UserMappingProfile>();
config.AssertConfigurationIsValid();
var mapper = new MappingEngine(config);
我可能会这样做:
var dict = new Dictionary<string, string> { { "Name", "Matt" }, { "Age", "27" } };
var user = mapper.Map<User, Dictionary<string, string>>(dict);
更简单的解决方案。 只需从 KeyValuePair 映射您的对象。 例子:
CreateMap<KeyValuePair<Guid, string>, User>()
.ForMember(u => u.Id, src => src.MapFrom(x => x.Key))
.ForMember(u => u.Name, src => src.MapFrom(x => x.Value));
如果您的函数类型是“ ExpandoObject ”,这将起作用。
public EmpClass
{
public string EmpName { get; set; }
public int EmpId { get; set; }
}
this.CreateMap<IDictionary<string, object>, EmpClass>()
.ForMember(dest => dest.EmpName, src => src.MapFrom(x => x["EmpName"]))
.ForMember(dest => dest.EmpId, src => src.MapFrom(x => x["EmpId"]));
如果有帮助,请告诉我。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.