使用AutoMapper,是否可以仅将更改的属性从视图模型映射到域对象?
我遇到的问题是,如果视图模型上有未更改的属性(null),那么它们将覆盖域对象并持久化到数据库。
是的,可以这样做,但是您必须在映射配置中指定何时使用Condition()
跳过目的地属性。
这里有一个例子。考虑以下类:
public class Source
{
public string Text { get; set; }
public bool Map { get; set; }
}
public class Destination
{
public string Text { get; set; }
}
第一个映射不会覆盖destination.Text
,但第二个映射会。
Mapper.CreateMap<Source, Destination>()
.ForMember(dest => dest.Text, opt => opt.Condition(src => src.Map));
var source = new Source { Text = "Do not map", Map = false };
var destination = new Destination { Text = "Leave me alone" };
Mapper.Map(source, destination);
source.Map = true;
var destination2 = new Destination { Text = "I'll be overwritten" };
Mapper.Map(source, destination2);
@Matthew Steven Monkan是正确的,但似乎AutoMapper改变了API。我把新的放上去供大家参考。
public static IMappingExpression<TSource, TDestination> MapOnlyIfChanged<TSource, TDestination>(this IMappingExpression<TSource, TDestination> map)
{
map.ForAllMembers(source =>
{
source.Condition((sourceObject, destObject, sourceProperty, destProperty) =>
{
if (sourceProperty == null)
return !(destProperty == null);
return !sourceProperty.Equals(destProperty);
});
});
return map;
}
对于Automapper版本<6.0
是的;我编写了这个扩展方法,只将脏值从模型映射到实体框架。
public static IMappingExpression<TSource, TDestination> MapOnlyIfDirty<TSource, TDestination>(
this IMappingExpression<TSource, TDestination> map)
{
map.ForAllMembers(source =>
{
source.Condition(resolutionContext =>
{
if (resolutionContext.SourceValue == null)
return !(resolutionContext.DestinationValue == null);
return !resolutionContext.SourceValue.Equals(resolutionContext.DestinationValue);
});
});
return map;
}
的例子:
Mapper.CreateMap<Model, Domain>().MapOnlyIfDirty();
No.
这正是您从不从视图模型映射到域模型的原因之一。领域/业务模型的更改太重要了,工具无法处理。
Automapper不是为处理这种情况而设计的:
AutoMapper工作是因为它强制了一个约定。它假设目标类型是源类型的子集。它假定目标类型上的所有内容都应该被映射。它对象的确切名称之后是目标成员名源类型。它假设您想要将复杂的模型扁平化为简单的。
从:https://jimmybogard.com/automappers-design-philosophy/
手动:
customer.LastName = viewModel.LastName
更改业务状态太重要了,否则无法执行