最近发现了一个比较有趣的东西 AutoMapper,主要将Model转换为DTO,DTO更注重数据,对领域对象进行合理封装,从而不会将领域对象的行为过度暴露给表现层。app
先来看一点实例,两个类之间的映射。ide
首先定义两个类Source与DTOSource:网站
1 public class Source 2 { 3 public int Id { get; set; } 4 public string Content { get; set; } 5 } 6 7 public class DTOSource 8 { 9 public int Id { get; set; } 10 public string Content { get; set; } 11 }
Source与DTOSource字段彻底相同,来看看它俩如何经过AutoMapper转换,代码很简单。spa
1 Mapper.Initialize(x=>{ 2 x.CreateMap<Source,DTOSource>(); 3 }); 4 5 Source s = new Source{Id=1,Content="123"}; 6 7 DTOSource dto = Mapper.Map<DTOSource>(s);
第一步创建Source到DTOSource之间的映射,初始化一个Source实例后,来看下执行结果:code
执行完成后,能够看到dto中的数据与以前初始化的s的数据是同样的,就像是直接将s拷贝了一份给dto,在两个类字段名定全相同的状况下如此,那么若是DTOSource中的字段名与Source中的不相同如何,其实也很简单,只需对象
要改为一点点的代码既可:blog
咱们将DTOSource中的Content的字段名改为Desc,此时只须要创建映射关系时,指定字段就能够了:继承
1 Mapper.Initialize(x => { 2 x.CreateMap<Source, DTOSource>().ForMember(c=>c.Desc,q=> { 3 q.MapFrom(z => z.Content); 4 }); 5 });
来看看运行结果如何;get
能够看到与以前的运行结果是相同的。string
那么如何映射两个List,其实也很简单,和上述代码几乎能够说是无差异,只是在最后一步时,要作一点点的修改就能够了。以下面代码:
1 Mapper.Initialize(x => { 2 x.CreateMap<Source, DTOSource>().ForMember(c => c.Desc, q => 3 { 4 q.MapFrom(z => z.Content); 5 }); 6 }); 7 8 s.Add(new Source { Id = 1, Content = "123" }); 9 10 var dto = Mapper.Map<List<DTOSource>>(s);
能够看到除了最后一句代码,其它几乎是彻底相同的,只是在最后一句代码中,目标类型改为了List<DTOSource>仅此而已。看下运行结果如何:
结果符合预期。
在实际的项目中,这样的写法确定是不符合要求的,通常会作一个封装,新建一个SourceProfile继承自Profile:
1 public SourceProfile() 2 { 3 base.CreateMap<Source, DTOSource>().ForMember(c => c.Desc, q => { 4 q.MapFrom(z => z.Content); 5 }); 6 }
全部映射关系均可以写在这一个类里,只须要在程序初始化的时候调用一次就能够了:
1 Mapper.Initialize(x => { x.AddProfile<SourceProfile>(); });
博主使用的AutoMapper版本6.1.1.0,由于AutoMapper在6.0版本时移除了Profile中的Configure,因此与6.0版本如下写法有点不一样,6.0如下版本写法为:
1 public class SourceProfile : Profile 2 { 3 protected override void Configure() 4 { 5 CreateMap<Source, DTOSource>().ForMember(c => c.Desc, q => { 6 q.MapFrom(z => z.Content); 7 }); 8 } 9 }
继承Profile重写其Configure便可,调用方式与上述没有太大差异。 Mapper.Initialize中可添加一个或多个Profile。
在MVC项目的应用中,能够将Mapper.Initialize封装到一个类里;
public static class AutoMapperForMvc { public static void Register() { Mapper.Initialize(x => { x.AddProfile<SourceProfile>(); }); } }
进而在MVC的Global中进一次性注册:
1 public class MvcApplication : System.Web.HttpApplication 2 { 3 protected void Application_Start() 4 { 5 AreaRegistration.RegisterAllAreas(); 6 FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); 7 RouteConfig.RegisterRoutes(RouteTable.Routes); 8 BundleConfig.RegisterBundles(BundleTable.Bundles); 9 //注册 10 AutoMapperForMvc.Register(); 11 } 12 }
更多有趣的东西能够查看官方网站