最近发现了一个比较有趣的东西 AutoMapper,主要将Model转换为DTO,DTO更注重数据,对领域对象进行合理封装,从而不会将领域对象的行为过分暴露给表现层。

先来看一点实例,两个类之前的映射。

首先定义两个类Source与DTOSource:

public class Source    {        public int Id { get; set; }        public string Content { get; set; }    }    public class DTOSource    {        public int Id { get; set; }        public string Content { get; set; }    }

Source与DTOSource字段完全相同,来看看它俩如何通过AutoMapper转换,代码很简单。

Mapper.Initialize(x=>{     x.CreateMap<Source,DTOSource>();});Source s = new Source{Id=1,Content="123"};DTOSource dto = Mapper.Map<DTOSource>(s);

第一步建立Source到DTOSource之间的映射,初始化一个Source实例后,来看下执行结果:

执行完成后,可以看到dto中的数据与之前初始化的s的数据是一样的,就像是直接将s拷贝了一份给dto,在两个类字段名定全相同的情况下如此,那么如果DTOSource中的字段名与Source中的不相同如何,其实也很简单,只需

要改成一点点的代码既可:

我们将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 });

来看看运行结果如何;

可以看到与之前的运行结果是相同的。

那么如何映射两个List,其实也很简单,和上述代码几乎可以说是无差别,只是在最后一步时,要做一点点的修改就可以了。如下面代码:

Mapper.Initialize(x => {                x.CreateMap<Source, DTOSource>().ForMember(c => c.Desc, q =>                {                    q.MapFrom(z => z.Content);                });            });            s.Add(new Source { Id = 1, Content = "123" });            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以下版本写法为:

public class SourceProfile : Profile    {        protected override void Configure()        {            CreateMap<Source, DTOSource>().ForMember(c => c.Desc, q => {                q.MapFrom(z => z.Content);            });        }    }

继承Profile重写其Configure即可,调用方式与上述没有太大差别。 Mapper.Initialize中可添加一个或多个Profile。

在MVC项目的应用中,可以将Mapper.Initialize封装到一个类里;

 public static class AutoMapperForMvc    {        public  static void Register()        {            Mapper.Initialize(x => {                x.AddProfile<SourceProfile>();            });        }    }

进而在MVC的Global中进一次性注册:

public class MvcApplication : System.Web.HttpApplication    {        protected void Application_Start()        {            AreaRegistration.RegisterAllAreas();            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);            RouteConfig.RegisterRoutes(RouteTable.Routes);            BundleConfig.RegisterBundles(BundleTable.Bundles);            //注册            AutoMapperForMvc.Register();        }    }

更多相关文章

  1. Asp.Net MVC实现分页、检索、排序的代码展示
  2. C#中使用反射以及特性简化的实例代码
  3. .Net实现微信JS-SDK分享功能代码展示
  4. C#中关于程序功能实现以及对代码选择的思考
  5. .net中关于异步性能测试的示例代码
  6. ASP.NET Core中用户登录验证实现最低配置的示例代码
  7. 代码详解AVL树的插入
  8. 厌倦写代码的人是如何做软件开发的
  9. 一招搞定C++调用Lua代码配置文件函数(附代码)

随机推荐

  1. AngularJS身份验证和基于XSRF令牌
  2. js 处理url中文参数 java端接收处理
  3. How to learn js properly(week4)使用js建
  4. jquery.fullCalendar官方文档翻译(一款小
  5. 转载和积累系列 - Nodejs HTTP多线程
  6. 在页面加载时在shadowbox中加载视频
  7. 原生JavaScript利用setInterval的一个简
  8. 结合悬停和点击功能(jQuery)?
  9. 初识javascript编程模式
  10. 即使通过一系列图像预先加载也表现不佳