先说说DTO

DTO是个什么东东?

DTO(Data Transfer Object)就是数据传输对象,说白了就是一个对象,只不过里边全是数据而已。

为什么要用DTO?

1、DTO更注重数据,对领域对象进行合理封装,从而不会将领域对象的行为过分暴露给表现层

2、DTO是面向UI的需求而设计的,而领域模型是面向业务而设计的。因此DTO更适合于和表现层的交互,通过DTO我们实现了表现层与领域Model之间的解耦,因此改动领域Model不会影响UI层

3、DTO说白了就是数据而已,不包含任何的业务逻辑,属于瘦身型的对象,使用时可以根据不同的UI需求进行灵活的运用

AutoMapper

现在我们既然知道了使用DTO的好处,那么我们肯定也想马上使用它,但是这里会牵扯一个问题:怎样实现DTO和领域Model之间的转换?

有两个思路,我们要么自己写转换代码,要么使用工具。不过就应用而言,我还是觉得用工具比较简单快捷,那就使用工具吧。其实这样的转换工具很多,不过我还是决定使用AutoMapper,因为它足够轻量级,而且也非常流行,国外的大牛们都使用它。使用AutoMapper可以很方便的实现DTO和领域Model之间的转换,它是一个强大的Object-Object Mapping工具。

摘自让AutoMapper在你的项目里飞一会儿

AutoMapper6.2.2.0

AutoMapper6.2.2.0与之前的版本有些不同,那么究竟有什么不同,我们一起来实践一下:

1.首先,使用AutoMapper6.2.2.0需要在你的项目中引用NuGet包,右键依赖项,管理NuGet程序包,然后选择浏览,搜索AutoMapper,安装,你就可以在项目中使用啦,你也可以在vs中使用打开工具-库程序包管理器-程序包管理控制平台,输入“Install-Package AutoMapper”命令,就可以把AutoMapper添加到项目中了~

2.让我们开始使用

  (1)一个简单的映射

      首先,创建一个C#控制台应用程序,为了方便,我们在Program直接定义三个类:

      

public class Student
{
public string Name { get; set; }
public int Sex { get; set; }
public string Age { get; set; }
public DateTime Birth { get; set; }
} public class Dto_Student
{
public string n { get; set; }
public string s { get; set; }
public int a { get; set; }
public string b { get; set; }
} public class V_Student
{
public string Name { get; set; }
public int Sex { get; set; }
public string Age { get; set; }
public DateTime Birth { get; set; }
}

  (2)一个简单的映射,由于Student和V_Student类字段名称一样,类型相同,所以,映射可以这么写

  

//一个简单的映射
AutoMapper.Mapper.Initialize(map => map.CreateMap<Student, V_Student>());
var stu = AutoMapper.Mapper.Map<Student>(new V_Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
}); var vstu = AutoMapper.Mapper.Map<V_Student>(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
});

  (3)那么,不同字段名称,甚至不同类型的两个类如何映射呢,那就要手动的映射相应字段了:

//属性不同名,属性类型不同映射
AutoMapper.Mapper.Initialize(map => map.CreateMap<Student, Dto_Student>()
.ForMember(d => d.n, opt =>
{
opt.MapFrom(s => s.Name);
})
.ForMember(d => d.s, opt =>
{
opt.MapFrom(s => s.Sex == 1 ? "男" : "女");
})
.ForMember(d => d.a, opt =>
{
opt.MapFrom(s => Convert.ToInt32(s.Age));
})
.ForMember(d => d.b, opt =>
{
opt.MapFrom(s => s.Birth.ToString("yyyy-MM-dd"));
})
); var dto_stu = AutoMapper.Mapper.Map<Dto_Student>(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
}); var stu = AutoMapper.Mapper.Map<Student>(new Dto_Student
{
n = "myname",
s = "男",
a = 24,
b = DateTime.Now.ToString("yyyy-MM-dd")
});

  这边一运行,发现报错了,为什么呢,原来是因为,由于映射字段类型不同,无法反向映射,那么,如何再添加一个映射呢,这时候就需要用到Profile这个类,我们需要继承这个类,并在里面写下映射配置,具体如下:

新建Dto_StudentProfile类和StudentProfile类:

public class Dto_StudentProfile:Profile
{
public Dto_StudentProfile()
{
base.CreateMap<Dto_Student, Student>()
.ForMember(s => s.Name, opt =>
{
opt.MapFrom(stu => stu.n);
})
.ForMember(s => s.Sex, opt =>
{
opt.MapFrom(stu => stu.s.Equals("男")?1:0);
})
.ForMember(s => s.Age, opt =>
{
opt.MapFrom(stu => stu.a.ToString());
})
.ForMember(s => s.Birth, opt =>
{
opt.MapFrom(stu =>DateTime.Parse( stu.b+" 00:00:00"));
});
}
} public class StudentProfile:Profile
{
public StudentProfile()
{
base.CreateMap<Student, Dto_Student>()
.ForMember(d => d.n, opt =>
{
opt.MapFrom(stu => stu.Name);
})
.ForMember(d => d.s, opt =>
{
opt.MapFrom(stu => stu.Sex == 1 ? "男" : "女");
})
.ForMember(d => d.a, opt =>
{
opt.MapFrom(stu => Convert.ToInt32(stu.Age));
})
.ForMember(d => d.b, opt =>
{
opt.MapFrom(stu => stu.Birth.ToString("yyyy-MM-dd"));
});
}
}

  那么,我们如何使用两个配置呢?

//配置映射
AutoMapper.Mapper.Initialize(map => map.AddProfiles(new[] {
typeof(Dto_StudentProfile),
typeof(StudentProfile)
}));
var dto_stu = AutoMapper.Mapper.Map<Dto_Student>(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
}); var stu = AutoMapper.Mapper.Map<Student>(new Dto_Student
{
n = "myname",
s = "男",
a = 24,
b = DateTime.Now.ToString("yyyy-MM-dd")
});

  这样,我们就完成了两个不同字段名称,不同类型的映射

(4)映射List<T>

  如何进行实体列表的映射呢,其实,配置上并没有任何不同,只需要在使用上,换成list就可以了:

List<Student> estu = new List<Student>();
estu.Add(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
});
estu.Add(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
});
List<Dto_Student> slist = AutoMapper.Mapper.Map<List<Student>, List<Dto_Student>>(estu);

  (5)你可能觉得不太方便,那么,我们可以将AutoMapper的Initialize放到应用开始的时候运行

  Mvc项目,可以放到程序的Global中,如果是.NET Core 2.0的MVC项目,可以放到StartUp中运行:

这边直接摘了前人的方法,有兴趣可以看一下https://www.cnblogs.com/lvlinlv/p/7344916.html

  (6)那么,你可以使用拓展的方法进行映射,这里定义一个AutoMapperHelper操作类:

using System;
using System.Collections;
using System.Collections.Generic;
using System.Text; namespace AutoMapperTest
{
public static class AutoMapperHelper
{
public static T MapTo<T>(this object obj)
{
if (obj == null) return default(T);
return AutoMapper.Mapper.Map<T>(obj);
} public static List<TDestination> MapToList<TDestination>(this object source)
{
return AutoMapper.Mapper.Map<List<TDestination>>(source);
}
}
}

  这样,你就可以这么使用:

List<Student> estu = new List<Student>();
estu.Add(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
});
estu.Add(new Student
{
Name = "myname",
Sex = 1,
Age = "24",
Birth = DateTime.Now
});
var slist = estu.MapToList<Dto_Student>();

  非list的实体映射也是一样的,这里不再多说,如果有什么疑问,欢迎提出来,共同讨论进步,感谢你的阅读

C#使用AutoMapper6.2.2.0进行对象映射的更多相关文章

  1. 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之十三 || DTOs 对象映射使用,项目部署Windows+Linux完整版

    更新 很多小伙伴在用 IIS 发布的时候,总是会有一些问题,文章下边 #autoid-6-0-0 我也简单的动图展示了,如何 publish 到 IIS 的过程,如果你能看懂,却发现自己的项目有问题的 ...

  2. php设计模式 数据对象映射模式

    数据对象映射模式,是将对象和数据存储映射起来,对一个对象的操作会映射为对数据存储的操作. 在代码中实现数据对象映射模式,实现一个ORM类,将复杂的sql语句映射成对象属性的操作.对象关系映射(Obje ...

  3. ASP.NET MVC 模型和数据对象映射实践

    在使用 MVC 开发项目的过程中遇到了个问题,就是模型和数据实体之间的如何快捷的转换?是不是可以像 Entity Framework 的那样 EntityTypeConfiguration,或者只需要 ...

  4. PHP 设计模式 笔记与总结(9)数据对象映射模式

    [数据对象映射模式] 是将对象和数据存储映射起来,对一个对象的操作会映射为对数据存储的操作.例如在代码中 new 一个对象,使用数据对象映射模式就可以将对象的一些操作比如设置一些属性,就会自动保存到数 ...

  5. 关于iOS上的对象映射公用方法-备

    具体的使用方法,请见下面说明,或者见工程里的单元测试代码.或者,参考原始文档: https://github.com/mystcolor/JTObjectMapping 使用方法 ======== 绝 ...

  6. MyBitis(iBitis)系列随笔之二:类型别名(typeAliases)与表-对象映射(ORM)

    类型别名(typeAliases):     作用:通过一个简单的别名来表示一个冗长的类型,这样可以降低复杂度.    类型别名标签typeAliases中可以包含多个typeAlias,如下 < ...

  7. rest-assured的对象映射(序列化和反序列化)

    rest-assured支持映射Java对象到Json和XML以及从Json和XML中映射到Java对象.Json映射需要在classpath 中有Jackson.Jackson 2或者是Gson,X ...

  8. 基于JsonPath和XmlPath的对象映射(Object Mapping)

    rest-assured支持映射Java对象到Json和XML以及从Json和XML中映射到Java对象.Json映射需要在classpath 中有Jackson.Jackson 2或者是Gson,X ...

  9. 一篇搞定SQLAlchemy--关系对象映射

    要使用SQLAlchemy,必须先下载这个模块 pip3 install sqlalchemy 或 pycharm File--> Settings-->project...-->P ...

随机推荐

  1. 实现Windows程序的更新

    实现Windows程序的更新 一.使用枚举避免不合理的赋值 1.使用枚举的好处: 使用常量类中Student类中加入一个特别属性,StudentGender,而且这个属性只能接受两个有效值," ...

  2. 使用ADO.NET查询和操作数据库

    String和StringBuilder 语法: //声明一个空的StringBuilder对象 StingBuilder对象名称 = new   StringBuilder(); //声明一个Str ...

  3. Online database documentation.

    贫道2018年1月正式出道,可以说在IT界我就是个菜鸟.但我有着一颗不服输的心,我相信我会在这条路走上巅峰之道的.下面我来写我的第一份学习笔记: 介绍:大多数公司都有自己的数据文档,估计大多数都是用P ...

  4. 关于Redis数据库 ---- 基础篇

    Redis数据库也被称为数据结构数据库,因为存储基于key-value模式. 其中,value值可以为字符串(string),哈希(map),列表(list),集合(set)和有序集合(zset). ...

  5. 【Java EE】从零开始写项目【总结】

    从零开发项目概述 最近这一直在复习数据结构和算法,也就是前面发出去的排序算法八大基础排序总结,Java实现单向链表,栈和队列就是这么简单,十道简单算法题等等... 被虐得不要不要的,即使是非常简单有时 ...

  6. Hive 报错:java.lang.RuntimeException: Unable to instantiate org.apache.hadoop.hive.metastore.HiveMetaStoreClient

    在配置好hive后启动报错信息如下: [walloce@bigdata-study- hive--cdh5.3.6]$ bin/hive Logging initialized using confi ...

  7. Redis——常见面试题

    一.memcached与redis的区别? 1.存储方式不同.memcached把数据全部存在内存之中,断电之后会挂掉,而redis虽然也用到了内存,但是会有部分数据存在硬盘中,保证数据持久性. 2. ...

  8. [福大软工] W班 软件产品案例分析

    作业要求 https://edu.cnblogs.com/campus/fzu/FZUSoftwareEngineering1715W/homework/1300 评分细则 第一部分 调研,评测 (3 ...

  9. DNS协议(一)

    在互联网上要想与另外一台主机通信,要知道对方的IP地址,但是IP地址是很难记忆的, 比如百度的一台服务器的IP地址为115.239.210.27,我们在浏览器中输入http://115.239.210 ...

  10. python day1 基本语法作业

    一.过7 start =1 while start<=10: if start !=7: print(start) start +=1 二.100以内的和 sum = 0 start = 1 w ...