Big Domainモデルをデータベースオブジェクトにマッピングするには、AutomApperを使用してください
-
02-10-2019 - |
質問
AutomApperを使用して、モデルオブジェクトをデータベースオブジェクトにマッピングしたいと思います。データベースオブジェクトが30を超えるフィールドであるとしましょう。モデルの20のプロパティのうち10個をマップしたいと考えています。それをより複雑にするには、データベースに新しいレコードを挿入するときよりも、レコードを更新するときに異なるプロパティをマップする必要があります。
私が使用しているソリューションは、両方のマッピングを指定する2つの一般的なクラスの挿入と更新およびマッピングプロファイルを作成することです。
以下の例:
public abstract class SyncMappingWrapper<TFrom> where TFrom : class
{
protected SyncMappingWrapper(TFrom model)
{
Model = model;
}
public TFrom Model { get; private set; }
}
public class Update<TFrom> : SyncMappingWrapper<TFrom> where TFrom : class
{
public Update(TFrom model)
: base(model)
{
}
}
public class Insert<TFrom> : SyncMappingWrapper<TFrom> where TFrom : class
{
public Insert(TFrom model)
: base(model)
{
}
}
ただし、サイクロマティックの複雑さが空(50を超える)に至るため、マッピングは厄介になっています。
CreateMap<Update<OracleModel>, LiveModel>()
.ForMember(des => des.ApprovedBy, opt => opt.Ignore())
.ForMember(des => des.ApprovedDate, opt => opt.Ignore())
...
.ForMember(des => des.UNSPSC, opt => opt.Ignore())
.ForMember(des => des.BaseUnit, opt => opt.MapFrom(src => src.Model.UOM.BaseUOM.PerSalesUnit))
.ForMember(des => des.BaseUOM, opt => opt.MapFrom(src => src.Model.UOM.BaseUOM.UnitOfMeasure.Code))
.ForMember(des => des.SalesUnit, opt => opt.MapFrom(src => src.Model.UOM.SalesUOM.PerSalesUnit))
.ForMember(des => des.SalesUOM, opt => opt.MapFrom(src => src.Model.UOM.SalesUOM.UnitOfMeasure.Code))
.ForMember(des => des.OrderUnit, opt => opt.MapFrom(src => src.Model.UOM.OrderUOM.PerSalesUnit))
.ForMember(des => des.OrderUOM, opt => opt.MapFrom(src => src.Model.UOM.OrderUOM.UnitOfMeasure.Code))
.ForMember(des => des.SalesPrice, opt => opt.MapFrom(src => src.Model.Price.Value))
.ForMember(des => des.Alternate, opt => opt.Ignore())
.ForMember(des => des.ManufacturerID, opt => opt.Ignore())
.ForMember(des => des.ProductCode, opt => opt.MapFrom(src => src.Model.ProductCode))
.ForMember(des => des.ProductName, opt => opt.MapFrom(src => src.Model.ProductName))
.ForMember(des => des.ProductHTML, opt => opt.Ignore())
.ForMember(des => des.Version, opt => opt.Ignore())
...
.ForMember(des => des.UnitsOfMeasure2, opt => opt.Ignore())
.ForMember(des => des.Manufacturer, opt => opt.Ignore());
新しいオブジェクトを作成することで、新しいレコードを挿入するための問題を解決しました。
CreateMap<Insert<OracleModel>, LiveModel>()
.ConstructUsing(x => new LiveModel
{
BaseUnit = x.Model.UOM.BaseUOM.PerSalesUnit,
BaseUOM = x.Model.UOM.BaseUOM.UnitOfMeasure.Code,
SalesUnit = x.Model.UOM.SalesUOM.PerSalesUnit,
SalesUOM = x.Model.UOM.SalesUOM.UnitOfMeasure.Code,
OrderUnit = x.Model.UOM.OrderUOM.PerSalesUnit,
OrderUOM = x.Model.UOM.OrderUOM.UnitOfMeasure.Code,
SalesPrice = x.Model.Price.Value,
LeadTime = x.Model.LeadTime,
ProductCode = x.Model.ProductCode,
ProductName = x.Model.ProductName,
SupplierCode = x.Model.SupplierCode,
Weight = x.Model.Weight
})
.ForAllMembers(xc => xc.Ignore());
ただし、新しいインスタンスではなく、既存のオブジェクトにプロパティをマッピングしたい更新では機能しません。
Mapper.Map(update, existingRecord);
マッピングを完全に制御するために、dynamicmap()を避けたいと思います(したがって、誤ってランダムプロパティをマップしません)。私の目標は、環状複雑さの問題を整理することです。 ValueInjecterまたはその他の方法を提案しないでください。 AutomApper内のソリューションを探しています。
解決
5年後ですが、このマッピングの複雑さの問題を軽減するための提案があります。最初にすべてのメンバーを無視するのに役立つ拡張機能を作成できます。
public static class AutoMapperExtension {
public static IMappingExpression<TSource, TDest> IgnoreAllMembers<TSource, TDest>(this IMappingExpression<TSource, TDest> expression) {
expression.ForAllMembers(opt => opt.Ignore());
return expression;
}
}
次に、実行するマッピングのみを定義するために使用します。
Mapper.CreateMap<Insert<OracleModel>, LiveModel>()
.IgnoreAllMembers()
.ForMember(d => d.BaseUnit, o => o.MapFrom(s => s.Model.UOM.BaseUOM.PerSalesUnit))
/* Mapping for other members here. */;
所属していません StackOverflow