安装方式:使用vs自带的nuget管理工具,搜索AutoMapper ,选择第一个安装到你的项目即可。

我从网上找了一些资料,

参考网址:http://blog.csdn.net/csethcrm/article/details/52934325

下载了个demo,然后自己又写了一遍,我把AutoMapper 的使用分为两种:

1、viewmodel与实体的字段名字是一致的,viewmodel的字段可以与实体中的字段数量不一致。

还有一种情况是:源实体中的字段名字是Getxxx,那么viewmodel中对应的字段可以是xxx,也会自动对应赋值,比如我写的demo中源实体中GetA,viewmodel中的A;

再有一种情况就是实体中的实体赋值,在我写的这个例子中,源实体中包含的实体类字段为Sub,里面包含的字段名字为Age,

那么destmodel中对应的字段名字可以是:SubAge,那么automapper就可以自动为你赋值了,大家看最后的运行结果。

给大家看下我建的源实体:

    public class Source1{public string Name { set; get; }public string GetA { set; get; }public string GetD { set; get; }public string SetB { set; get; }public string c { set; get; }public SubSource1 Sub { set; get; }}public class SubSource1{public string Age { set; get; }}

还有viewmodel(要转化成为你想要的模型):

    public class Dest1{public string Name { set; get; }public string A { set; get; }public string C { set; get; }public string SubAge { set; get; }public string D { set; get; }}

我封装的扩展方法:

        /// <summary>///  类型映射,默认字段名字一一对应/// </summary>/// <typeparam name="TDestination">转化之后的model,可以理解为viewmodel</typeparam>/// <typeparam name="TSource">要被转化的实体,Entity</typeparam>/// <param name="source">可以使用这个扩展方法的类型,任何引用类型</param>/// <returns>转化之后的实体</returns>public static TDestination MapTo<TDestination, TSource>(this TSource source)where TDestination : classwhere TSource : class{if (source == null) return default(TDestination);var config = new MapperConfiguration(cfg => cfg.CreateMap<TSource, TDestination>());var mapper = config.CreateMapper();return mapper.Map<TDestination>(source);}

使用方式:

                var source1 = new Source1{Name = "source",Sub = new SubSource1 { Age = "25" },c = "c",GetA = "A",SetB = "B"};var destViewModel = source1.MapTo<Source1,Dest1>();

运行结果:

2.viewmodel与实体字段名字没有全部对应,只有几个字段的名字和源实体中的字段名字是一样的,其他的字段是通过实体中的几个字段组合或者是格式或者是类型转化而来的,

使用方法:不能再使用这个扩展方法了,只能自己额外写代码,代码如下:

               var config2 = new MapperConfiguration(cfg => cfg.CreateMap<SourceUser, DestUser2>().ForMember(d => d.DestName, opt => opt.MapFrom(s => s.Name))    //指定字段一一对应.ForMember(d => d.Birthday, opt => opt.MapFrom(src => src.Birthday.ToString("yy-MM-dd HH:mm")))//指定字段,并转化指定的格式.ForMember(d => d.Age, opt => opt.Condition(src => src.Age > 5))//条件赋值.ForMember(d => d.A1, opt => opt.Ignore())//忽略该字段,不给该字段赋值.ForMember(d => d.A1, opt => opt.NullSubstitute("Default Value"))//如果源字段值为空,则赋值为 Default Value.ForMember(d => d.A1, opt => opt.MapFrom(src => src.Name + src.Age * 3 + src.Birthday.ToString("d"))));//可以自己随意组合赋值var mapper2 = config2.CreateMapper();

注释中都包含了平时常用的几种情况,其他的我就没有再写。

下面再给大家把list转化的扩展方法代码贴上:

        /// <summary>/// 集合列表类型映射,默认字段名字一一对应/// </summary>/// <typeparam name="TDestination">转化之后的model,可以理解为viewmodel</typeparam>/// <typeparam name="TSource">要被转化的实体,Entity</typeparam>/// <param name="source">可以使用这个扩展方法的类型,任何引用类型</param>/// <returns>转化之后的实体列表</returns>public static IEnumerable<TDestination> MapToList<TSource,TDestination>(this IEnumerable<TSource> source)where TDestination : classwhere TSource : class{if (source == null) return new List<TDestination>();var config = new MapperConfiguration(cfg => cfg.CreateMap<TSource, TDestination>());var mapper = config.CreateMapper();return mapper.Map<List<TDestination>>(source);}

同样的使用方式:

                var source1 = new Source1{Name = "source",Sub = new SubSource1 { Age = "25" },c = "c",GetA = "A",SetB = "B"};var source3 = new Source1{Name = "source3",Sub = new SubSource1 { Age = "253" },c = "c3",GetA = "A3",SetB = "B3"};var sourceList = new List<Source1> { source1, source3 };var destViewModelList = sourceList.MapToList<Source1,Dest1>();

运行结果:

以上就是我个人所得,如有错误,欢迎大家指正。

//2017.12.4 修改:destination和source写反了,导致我的总结有些错误,现在纠正一下:错误结论已经红色标注,中间的截图也换成正确的了,工具类方法也已经修正。

出处:https://www.cnblogs.com/dawenyang/p/7966850.html

===========================================================

一、最简单的用法

有两个类User和UserDto

 1     public class User
 2     {
 3         public int Id { get; set; }
 4         public string Name { get; set; }
 5         public int Age { get; set; }
 6     }
 7
 8     public class UserDto
 9     {
10         public string Name { get; set; }
11         public int Age { get; set; }
12     }

将User转换成UserDto也和简单

1     Mapper.Initialize(x => x.CreateMap<User, UserDto>());
2     User user = new User()
3     {
4         Id = 1,
5         Name = "caoyc",
6         Age = 20
7     };
8     var dto = Mapper.Map<UserDto>(user);

这是一种最简单的使用,AutoMapper会更加字段名称去自动对于,忽略大小写。

二、如果属性名称不同

将UserDto的Name属性改成Name2

 1     Mapper.Initialize(x =>
 2         x.CreateMap<User, UserDto>()
 3          .ForMember(d =>d.Name2, opt => {
 4             opt.MapFrom(s => s.Name);
 5             })
 6         );
 7
 8     User user = new User()
 9     {
10         Id = 1,
11         Name = "caoyc",
12         Age = 20
13     };
14
15     var dto = Mapper.Map<UserDto>(user);

三、使用Profile配置

自定义一个UserProfile类继承Profile,并重写Configure方法

 1     public class UserProfile : Profile
 2     {
 3         protected override void Configure()
 4         {
 5             CreateMap<User, UserDto>()
 6                 .ForMember(d => d.Name2, opt =>
 7                 {
 8                     opt.MapFrom(s => s.Name);
 9                 });
10         }
11     }

使用时就这样

 1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
 2
 3     User user = new User()
 4     {
 5         Id = 1,
 6         Name = "caoyc",
 7         Age = 20
 8     };
 9
10     var dto = Mapper.Map<UserDto>(user);

四、空值替换NullSubstitute

空值替换允许我们将Source对象中的空值在转换为Destination的值的时候,使用指定的值来替换空值。

 1     public class UserProfile : Profile
 2     {
 3         protected override void Configure()
 4         {
 5             CreateMap<User, UserDto>()
 6                 .ForMember(d => d.Name2, opt => opt.MapFrom(s => s.Name))
 7                 .ForMember(d => d.Name2, opt => opt.NullSubstitute("值为空"));
 8
 9         }
10     }

1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
2
3     User user = new User()
4     {
5         Id = 1,
6         Age = 20
7     };
8
9     var dto = Mapper.Map<UserDto>(user);

结果为:

五、忽略属性Ignore

 1     public class User
 2     {
 3         public int Id { get; set; }
 4         public string Name { get; set; }
 5         public int Age { get; set; }
 6     }
 7
 8     public class UserDto
 9     {
10         public string Name { get; set; }
11         public int Age { get; set; }
12
13     }
14
15     public class UserProfile : Profile
16     {
17         protected override void Configure()
18         {
19             CreateMap<User, UserDto>().ForMember("Name", opt => opt.Ignore());
20         }
21     }

使用

 1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
 2
 3     User user = new User()
 4     {
 5         Id = 1,
 6         Name="caoyc",
 7         Age = 20
 8     };
 9
10     var dto = Mapper.Map<UserDto>(user);

结果:

六、预设值

如果目标属性多于源属性,可以进行预设值

 1     public class User
 2     {
 3         public int Id { get; set; }
 4         public string Name { get; set; }
 5         public int Age { get; set; }
 6     }
 7
 8     public class UserDto
 9     {
10         public string Name { get; set; }
11         public int Age { get; set; }
12         public string Gender { get; set; }
13
14     }
15
16     public class UserProfile : Profile
17     {
18         protected override void Configure()
19         {
20             CreateMap<User, UserDto>();
21         }
22     }

使用

 1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
 2
 3     User user = new User()
 4     {
 5         Id = 1,
 6         Name="caoyc",
 7         Age = 20
 8     };
 9
10     UserDto dto = new UserDto() {Gender = "男"};
11     Mapper.Map(user, dto);

七、类型转换ITypeConverter

如果数据中Gender存储的int类型,而DTO中Gender是String类型

1     public class User
2     {
3         public int Gender { get; set; }
4     }
5
6     public class UserDto
7     {
8         public string Gender { get; set; }
9     }

类型转换类,需要实现接口ITypeConverter

 1     public class GenderTypeConvertert : ITypeConverter<int, string>
 2     {
 3         public string Convert(int source, string destination, ResolutionContext context)
 4         {
 5             switch (source)
 6             {
 7                 case 0:
 8                     destination = "男";
 9                     break;
10                 case 1:
11                     destination = "女";
12                     break;
13                 default:
14                     destination = "未知";
15                     break;
16             }
17             return destination;
18         }
19     }

配置规则

 1     public class UserProfile : Profile
 2     {
 3         protected override void Configure()
 4         {
 5             CreateMap<User, UserDto>();
 6
 7             CreateMap<int, string>().ConvertUsing<GenderTypeConvertert>();
 8             //也可以写这样
 9             //CreateMap<int, string>().ConvertUsing(new GenderTypeConvertert());
10         }
11     }

使用

 1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
 2
 3     User user0 = new User() { Gender = 0 };
 4     User user1 = new User() { Gender = 1 };
 5     User user2 = new User() { Gender = 2 };
 6     var dto0= Mapper.Map<UserDto>(user0);
 7     var dto1 = Mapper.Map<UserDto>(user1);
 8     var dto2 = Mapper.Map<UserDto>(user2);
 9
10     Console.WriteLine("dto0:{0}", dto0.Gender);
11     Console.WriteLine("dto1:{0}", dto1.Gender);
12     Console.WriteLine("dto2:{0}", dto2.Gender);

结果

八、条件约束Condition

当满足条件时才进行映射字段,例如人类年龄,假设我们现在人类年龄范围为0-200岁(这只是假设),只有满足在这个条件才进行映射

DTO和Entity

1     public class User
2     {
3         public int Age { get; set; }
4     }
5
6     public class UserDto
7     {
8         public int Age { get; set; }
9     }

Profile

1     public class UserProfile : Profile
2     {
3         protected override void Configure()
4         {
5             CreateMap<User, UserDto>().ForMember(dest=>dest.Age,opt=>opt.Condition(src=>src.Age>=0 && src.Age<=200));
6         }
7     }

使用代码

 1     Mapper.Initialize(x => x.AddProfile<UserProfile>());
 2
 3     User user0 = new User() { Age = 1 };
 4     User user1 = new User() { Age = 150 };
 5     User user2 = new User() { Age = 201 };
 6     var dto0= Mapper.Map<UserDto>(user0);
 7     var dto1 = Mapper.Map<UserDto>(user1);
 8     var dto2 = Mapper.Map<UserDto>(user2);
 9
10     Console.WriteLine("dto0:{0}", dto0.Age);
11     Console.WriteLine("dto1:{0}", dto1.Age);
12     Console.WriteLine("dto2:{0}", dto2.Age);

输出结果

出处:https://www.cnblogs.com/caoyc/p/6367828.html

===================================================

AutoMapper介绍

为什么要使用AutoMapper?

我们在实现两个实体之间的转换,首先想到的就是新的一个对象,这个实体的字段等于另一个实体的字段,这样确实能够实现两个实体之间的转换,但这种方式的扩展性,灵活性非常差,维护起来相当麻烦;实体之前转换的工具有很多,不过我还是决定使用AutoMapper,因为它足够轻量级,而且也非常流行,国外的大牛们都使用它使用AutoMapper可以很方便的实现实体和实体之间的转换,它是一个强大的对象映射工具。

一,如何添加AutoMapper到项目中?

在vs中使用打开工具 - 库程序包管理器 - 程序包管理控制平台,输入“Install-Package AutoMapper”命令,就可以把AutoMapper添加到项目中了〜

二,举个栗子

栗子1:两个实体之间的映射

Mapper.CreateMap <Test1,Test2>(); Test1 test1 = new Test1 {Id = 1,Name =“张三”,Date = DateTime.Now}; Test2 test2 = Mapper.Map <Test2>(test1);

栗子2:两个实体不同字段之间的映射

Mapper.CreateMap <Test1,Test2>()。ForMember(d => d.Name121,opt => opt.MapFrom(s => s.Name));

栗子3:泛型之间的映射

  1. Mapper.CreateMap <Test1,Test2>(); 
  2. var testList = Mapper.Map <List <Test1>,List <Test2 >>(testList);

三,扩展映射方法使映射变得更简单

  1. using System.Collections;
  2. using System.Collections.Generic;
  3. using System.Data;
  4. using AutoMapper;
  5. namespace Infrastructure.Utility
  6. {
  7. /// <summary>
  8. /// AutoMapper扩展帮助类
  9. /// </summary>
  10. public static class AutoMapperHelper
  11. {
  12. /// <summary>
  13. /// 类型映射
  14. /// </summary>
  15. public static T MapTo<T>(this object obj)
  16. {
  17. if (obj == null) return default(T);
  18. Mapper.CreateMap(obj.GetType(), typeof(T));
  19. return Mapper.Map<T>(obj);
  20. }
  21. /// <summary>
  22. /// 集合列表类型映射
  23. /// </summary>
  24. public static List<TDestination> MapToList<TDestination>(this IEnumerable source)
  25. {
  26. foreach (var first in source)
  27. {
  28. var type = first.GetType();
  29. Mapper.CreateMap(type, typeof(TDestination));
  30. break;
  31. }
  32. return Mapper.Map<List<TDestination>>(source);
  33. }
  34. /// <summary>
  35. /// 集合列表类型映射
  36. /// </summary>
  37. public static List<TDestination> MapToList<TSource, TDestination>(this IEnumerable<TSource> source)
  38. {
  39. //IEnumerable<T> 类型需要创建元素的映射
  40. Mapper.CreateMap<TSource, TDestination>();
  41. return Mapper.Map<List<TDestination>>(source);
  42. }
  43. /// <summary>
  44. /// 类型映射
  45. /// </summary>
  46. public static TDestination MapTo<TSource, TDestination>(this TSource source, TDestination destination)
  47. where TSource : class
  48. where TDestination : class
  49. {
  50. if (source == null) return destination;
  51. Mapper.CreateMap<TSource, TDestination>();
  52. return Mapper.Map(source, destination);
  53. }
  54. /// <summary>
  55. /// DataReader映射
  56. /// </summary>
  57. public static IEnumerable<T> DataReaderMapTo<T>(this IDataReader reader)
  58. {
  59. Mapper.Reset();
  60. Mapper.CreateMap<IDataReader, IEnumerable<T>>();
  61. return Mapper.Map<IDataReader, IEnumerable<T>>(reader);
  62. }
  63. }
  64. }

这样的话,你就可以这样使用了

var testDto = test.MapTo <Test2>();

var testDtoList = testList.MapTo <Test2>();

出处:https://blog.csdn.net/qq_35193189/article/details/80805451

=========================================================

根据上面的理解和站在巨人的肩膀上,自己重新封装一次,用到项目中。

    public static class AutoMapHelper{/// <summary>///  类型映射,默认字段名字一一对应/// </summary>/// <typeparam name="TDestination"></typeparam>/// <param name="obj"></param>/// <returns></returns>public static TDestination AutoMapTo<TDestination>(this object obj){if (obj == null) return default(TDestination);var config = new AutoMapper.MapperConfiguration(cfg => cfg.CreateMap(obj.GetType(), typeof(TDestination)));return config.CreateMapper().Map<TDestination>(obj);}/// <summary>/// 类型映射,可指定映射字段的配置信息/// </summary>/// <typeparam name="TSource">源数据:要被转化的实体对象</typeparam>/// <typeparam name="TDestination">目标数据:转换后的实体对象</typeparam>/// <param name="source">任何引用类型对象</param>/// <param name="cfgExp">可为null,则自动一一映射</param>/// <returns></returns>public static TDestination AutoMapTo<TSource, TDestination>(this TSource source, Action<AutoMapper.IMapperConfigurationExpression> cfgExp)where TDestination : classwhere TSource : class{if (source == null) return default(TDestination);var config = new AutoMapper.MapperConfiguration(cfgExp != null ? cfgExp : cfg => cfg.CreateMap<TSource, TDestination>());var mapper = config.CreateMapper();return mapper.Map<TDestination>(source);}/// <summary>/// 类型映射,默认字段名字一一对应/// </summary>/// <typeparam name="TSource">源数据:要被转化的实体对象</typeparam>/// <typeparam name="TDestination">目标数据:转换后的实体对象</typeparam>/// <param name="source">任何引用类型对象</param>/// <returns>转化之后的实体</returns>public static TDestination AutoMapTo<TSource, TDestination>(this TSource source)where TDestination : classwhere TSource : class{if (source == null) return default(TDestination);var config = new AutoMapper.MapperConfiguration(cfg => cfg.CreateMap<TSource, TDestination>());var mapper = config.CreateMapper();return mapper.Map<TDestination>(source);}/// <summary>/// 集合列表类型映射,默认字段名字一一对应/// </summary>/// <typeparam name="TDestination">转化之后的实体对象,可以理解为viewmodel</typeparam>/// <typeparam name="TSource">要被转化的实体对象,Entity</typeparam>/// <param name="source">通过泛型指定的这个扩展方法的类型,理论任何引用类型</param>/// <returns>转化之后的实体列表</returns>public static IEnumerable<TDestination> AutoMapTo<TSource, TDestination>(this IEnumerable<TSource> source)where TDestination : classwhere TSource : class{if (source == null) return new List<TDestination>();var config = new AutoMapper.MapperConfiguration(cfg => cfg.CreateMap<TSource, TDestination>());var mapper = config.CreateMapper();return mapper.Map<List<TDestination>>(source);}}

View Code

其实还可以继续优化,在泛型中不需要知道指定source的类型了,因为可以直接获取到,后面有时间再调整吧。

转载于:https://www.cnblogs.com/mq0036/p/10670202.html

c# AutoMapper 使用方式和再封装相关推荐

  1. httpurlconnection 封装_不要再封装各种Util工具类了,看看这个框架

    不要再封装各种Util工具类了,看看这个框架 Hutool 谐音 "糊涂",寓意追求 "万事都作糊涂观,无所谓失,无所谓得" 的境界.    Hutool 是一 ...

  2. 不要再封装各种Util工具类了,这个神级框架值得拥有!

    点击上方蓝色"方志朋",选择"设为星标" 回复"666"获取独家整理的学习资料! Hutool 谐音 "糊涂",寓意追求 ...

  3. Selenium4.0+Python手撸自动化框架系列之 Web元素等待方式介绍 与 封装

    目录 前言 三种等待 一.线程等待 二.隐性等待 三.显性等待 封装 一.参数设计 二.函数名设计 三.封装代码设计 前言 web自动化测试,常常因为硬件配制,浏览器,网速等因素导致网页加载速度过慢, ...

  4. ajax封装:get封装方式、post封装方式

    ajax调用参数解释: ajax({url:"http://localhost:3000/users",method:"POST",data:"use ...

  5. 小米推送php文档,利用php重载和curl的并行方式优雅的封装小米推送sdk

    前段时间迁移小米推送部分代码,这部分之前是其他人负责的.读了代码,发现了两点: 所有接口的实现除了url和传参基本都是一致的 android和ios的设备需要分别推送一次 刚好这段时间了解了一下php ...

  6. Ajax的get/post两种请求方式及ajax封装函数

    AJAX基本用法 引言 Ajax是浏览器提供的一套方法,可以用来实现页面无刷新更新数据,提高用户体验.主要有GET和POST两种请求方式. 1. GET请求处理 GET请求会将数据放到URL后面 GE ...

  7. 加密封装 怎么把_不要再封装各种Util工具类了,这个神级框架值得拥有!

    Hutool 谐音 "糊涂",寓意追求 "万事都作糊涂观,无所谓失,无所谓得" 的境界. Hutool 是一个 Java 工具包,也只是一个工具包,它帮助我们简 ...

  8. runtime Caller 方法再封装

    在 Go单元测试--资源初始化 里获取配置文件的绝对路径时,提到了 Caller 方法.这个方法在日志打印的包中特别常见. 日志文件中记录的文件路径和行号,就是这个函数的返回值. pc, file, ...

  9. log4c日志库的用法,再封装及其完整例子

    找了多篇文章,发现都讲的不怎么简明.自己摸索了下,记录如下: Log4c中有三个重要的概念, Category, Appender, Layout. Category用于区分不同的Logger, 其实 ...

最新文章

  1. silverlight 打开html_Silverlight与HTML集成方法
  2. Android --- 新版本取色器吸管工具点击无效,没反应
  3. oracle sql文本 参数,ORACLE 参数文件
  4. 微视已死,腾讯战略放弃微视,大牛纷纷离职,PMcaff--行业内部解读
  5. 专栏 | 基于 Jupyter 的特征工程手册:数据预处理(三)
  6. CTF——Web——MD5漏洞
  7. 2019计算机国二操作题,2019年3月计算机二级C++操作练习题及答案(十二)
  8. Qt控件总结:QToolButton
  9. php 字节码查看,PHP字节码缓存和内置服务器
  10. 使用socket方式连接Nginx优化php-fpm性能
  11. Sql优化总结!详细!(2021最新面试必问)
  12. 三极管当做开关的导通方式
  13. 野火指南者开发板移植 lvgl 库
  14. win10任务管理器禁用_如何在Windows 10的文件资源管理器中禁用广告和通知
  15. arduino定时器控制舵机_【求教】如何控制舵机的速度?
  16. 【训练1】储蓄卡与信用卡
  17. 安卓手机+python基于abd命令的自动打卡
  18. 前端Photoshop切图快速入门
  19. 会计科目 分类 说明
  20. 1646. Prime Path

热门文章

  1. 【Android】动态更新Icon
  2. 分析师分析业务维度,(个人制作分析思维导图Xmind)
  3. c windows控制台输出颜色文字
  4. 一道小时候经常玩的数字游戏
  5. Solr安装(单机版)
  6. AC日记——字符替换 openjudge 1.7 08
  7. Discuz!X安装配置
  8. Aspose Cells 控件如何实现数据过滤(附代码和下载地址)
  9. oracle 表导入到powerDesigner 中
  10. Ubuntu 18.04 固定ip 设置方法