AutoMapper在C#中的有趣應用


最近發現了一個比較有趣的東西 AutoMapper,主要將Model轉換為DTO,DTO更注重數據,對領域對象進行合理封裝,從而不會將領域對象的行為過分暴露給表現層。

先來看一點實例,兩個類之間的映射。

首先定義兩個類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轉換,代碼很簡單。

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實例后,來看下執行結果:

執行完成后,可以看到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,其實也很簡單,和上述代碼幾乎可以說是無差別,只是在最后一步時,要做一點點的修改就可以了。如下面代碼:

 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     }

 更多有趣的東西可以查看官方網站


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM