在实体与DTO之间,我们一般都需要进行映射。如果手动的来进行转换,实在是太麻烦。所以就产生了很多映射工具,比如AutoMapper,EmitMapper。而经过一些对比,EmitMapper比较快,接近于手工转换的速度。

          EmitMapper使用非常简单,对于那种属性名一样的,直接使用默认映射即可。如:

ObjectsMapper<From,To> mapper = ObjectMapperManager.DefaultInstance.GetMapper<From,To>();

to = mapper.Map(from);

        它就会自动的把From对象的值赋给To对象;

       对于有一定规则的,可以使用一些扩展的方法,如:

      通过使用ConvertUsing可以配置一些组合映射.比如这个字段可以是某两个字段拼接,或者其他一些计算.具体可以参见一些使用例子:

       http://www.cnblogs.com/aaa6818162/archive/2012/06/21/2557879.html

      http://www.cnblogs.com/wuhong/archive/2011/09/21/2184313.html

          但是在使用的过程中,发现了一个问题,就是本身默认是如果属性名相同就直接映射,而使用ConvertUsing是可以自定义映射条件的.但是如果使用了ConvertUsing,那么你必需为里面每个字段都指定映射的条件.而其实真正想用的是,在实体与DTO之间,其实大部分字段的属性名称是相同的,对于这些字段,我们是想直接默认映射,而对于其他一些需要自定义映射的字段,来自定义映射规则.但实际上却达不到这种效果,一般人也不想去把每个字段的映射规则配置一下,如果这样,还不如手动写呢.

          后来在网上找到了一个例子,可以使用FlatteringConfig这个自定义配置,这个类在EmitMapper的源码中是有的,但是不是在当前的解决方案下,而是在一个叫EMConfigurations的项目里.可以这样使用这个自定义配置.

public class User

{

  public Guid Id { get; set; }

  public Company Company { get; set; }

}    

 

public class Company

{

  public Guid Id { get; set; }

}

 

public class UserDTO

{

  public Guid Id { get; set; }

  public Guid CompanyId{ get; set; }

}

ObjectMapperManager.DefaultInstance.GetMapper<User, UserDTO>(

                new FlatteringConfig()

            );

var dto = mapper.Map(new User());

          这样,它就会自动的给DTO中的CompanyId赋值,也会给Id赋值.不过有一个规则就是CompanyId的命名要有一定的规则,一定要是那个实体的名称再加上这个实体里面属性的名称.这样才能进行自动的映射.(不过这时,如果那个实体为null,那也有可能会报错,没有试过,后面有时间试一下).

          在运行的过程中,发现了FlatteringConfig类的一个bug;这个类的原始代码如下:

public class FlatteringConfig : DefaultMapConfig 

    {

        protected Func<string, string, bool> nestedMembersMatcher;

 

        public FlatteringConfig()

        {

            nestedMembersMatcher = (m1, m2) => m1.StartsWith(m2);

        }

 

        public override IMappingOperation[] GetMappingOperations(Type from, Type to)

        {

            var destinationMembers = GetDestinationMemebers(to);

            var sourceMembers = GetSourceMemebers(from);

            var result = new List<IMappingOperation>();

            foreach (var dest in destinationMembers)

            {

                var matchedChain = GetMatchedChain(dest.Name, sourceMembers).ToArray();

                if (matchedChain == null || matchedChain.Length == 0)

                {

                    continue;

                }

                result.Add(

                    new ReadWriteSimple

                    {

                        Source = new MemberDescriptor(matchedChain),

                        Destination = new MemberDescriptor(new[] { dest })

                    }

                );

            }

            return result.ToArray();

        }

 

        public DefaultMapConfig MatchNestedMembers(Func<string, string, bool> nestedMembersMatcher)

        {

            this.nestedMembersMatcher = nestedMembersMatcher;

            return this;

        }

 

        private List<MemberInfo> GetMatchedChain(string destName, List<MemberInfo> sourceMembers)

        {

            var matches = sourceMembers.Where(s => MatchMembers(destName, s.Name) || nestedMembersMatcher(destName, s.Name));

            int len = 0;

            MemberInfo match = null;

            foreach (var m in matches)

            {

                if (m.Name.Length > len)

                {

                    len = m.Name.Length;

                    match = m;

                }

            }

            if (match == null)

            {

                return null;

            }

            var result = new List<MemberInfo> { match };

            if (!MatchMembers(destName, match.Name))

            {

                result.AddRange(

                    GetMatchedChain(destName.Substring(match.Name.Length), GetDestinationMemebers(match))

                );

            }

            return result;

        }

 

        private static List<MemberInfo> GetSourceMemebers(Type t)

        {

            return GetMemebers(t)

                .Where(

                    m => 

                        m.MemberType == MemberTypes.Field || 

                        m.MemberType == MemberTypes.Property ||

                        m.MemberType == MemberTypes.Method

                )

                .ToList();

        }

 

        private static List<MemberInfo> GetDestinationMemebers(MemberInfo mi)

        {

            Type t;

            if (mi.MemberType == MemberTypes.Field)

            {

                t = mi.DeclaringType.GetField(mi.Name).FieldType;

            }

            else

            {

                t = mi.DeclaringType.GetProperty(mi.Name).PropertyType;

            }

            return GetDestinationMemebers(t);

        }

 

        private static List<MemberInfo> GetDestinationMemebers(Type t)

        {

            return GetMemebers(t).Where(m => m.MemberType == MemberTypes.Field || m.MemberType == MemberTypes.Property).ToList();

        }

 

        private static List<MemberInfo> GetMemebers(Type t)

        {

            BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.Public;

            return t.GetMembers(bindingFlags).ToList();

        }

    }

      在GetMappingOperations方法中,修改了以下代码:(原代码中是一个判断在一起,改成分作两个判断)

var matchedChainSingle = GetMatchedChain(dest.Name, sourceMembers);

               if (matchedChainSingle == null)

               {

                   continue;

               }

               var matchedChain = matchedChainSingle.ToArray();

               if (matchedChain.Length == 0)

               {

                   continue;

               }

 

      参考资料:

      http://emitmapper.codeplex.com/

      http://stackoverflow.com/questions/9619265/emit-mapper-flattering-and-property-name-mismatch

      http://emitmapper.codeplex.com/SourceControl/changeset/view/69894#1192663

      http://emitmapper.codeplex.com/SourceControl/changeset/view/42128#691132

     http://stackoverflow.com/questions/12542749/emitmapper-flattering-config-nullreferenceexception?rq=1

EmitMapper自动映射工具的更多相关文章

  1. AutoMapper 自动映射工具

    先引用对应的DLL. 11.转换匿名对象 结合LINQ映射新的实体类. using System;using System.Collections.Generic;using System.Linq; ...

  2. 使用SpringDataJdbc的@Query注解实现自动映射结果集 ----- RowMapper接口

    使用@Query注解的时候,常常需要写sql来映射非域类的实例,通常的做法就是 实现 RowMapper接口,然后new实例一个一个的设置值进去...为此.出世了自动映射工具类 注意事项:此抽象类只是 ...

  3. [转]p2p端口映射工具 dog-tunnel

    [转]p2p端口映射工具 dog-tunnel http://www.oschina.net/p/dog-tunnel 狗洞是一个高速的 P2P 端口映射工具,同时支持Socks5代理. 0.5版后开 ...

  4. 代码自动生成工具MyGeneration之一(程序员必备工具)

    代码自动生成工具MyGeneration之一(程序员必备工具) 转 分类: C#2008-08-06 18:12 16064人阅读 评论(12) 收藏 举报 工具数据库相关数据库stringbrows ...

  5. .NET的DTO映射工具AutoMapper

    .NET的DTO映射工具AutoMapper 原文:https://github.com/AutoMapper/AutoMapper/wiki/Getting-started 参考:http://ww ...

  6. 反射+自定义注解---实现Excel数据列属性和JavaBean属性的自动映射

    简单粗暴,直奔主题.   需求:通过自定义注解和反射技术,将Excel文件中的数据自动映射到pojo类中,最终返回一个List<pojo>集合? 今天我只是通过一位使用者的身份来给各位分享 ...

  7. Touch 方法&属性 映射工具

    Touch 方法&属性 映射工具(0.5 版本) 标签 : github 线上后门与接口调试: 原先需要测试一个接口(如Dubbo.DAO), 或为线上留后门, 需要写大量的Web层(Api. ...

  8. AutoMapper自动映射

    十年河东,十年河西,莫欺少年穷. 学无止境,精益求精. 不扯犊子,直接进入正题: AutoMapper自动映射常用于EF中,能很好的解决DTO和Model之间相互映射的问题.在未使用AutoMappe ...

  9. 常见Bean映射工具分析评测及Orika介绍

    原地址:http://tech.dianwoda.com/2017/11/04/gao-xing-neng-te-xing-feng-fu-de-beanying-she-gong-ju-orika/ ...

随机推荐

  1. 跨路径读取cookie

    同域下,即使设置了cookie的路径还是能将不同路径cookie读出来. 1.在/ctf/day3/ 目录设置一个cookie 2.其他目录下是不能访问这个cookie的 3.通过iframe可以实现 ...

  2. VIM在Win7上的安装教程

    1.下载 目前VIM在其官网上的最新版本为7.4,Windows版本名称为GVIM,在百度软件中可以下载GVIM的最新版本,建议 在百度上下载,因为比较快.在百度上搜索"GVIM" ...

  3. mixup: Beyond Empirical Risk Minimization

    这篇论文MIT和FAIR的工作,主要是提出了一种mixup的方式.(感觉是一种产生hard sample的方法,是一种新的.更有效的数据增强.) 1 Introduction 大网络需要大数据,目前C ...

  4. 每天to do list

    至少写一页书 写代码做一个实验 读10+页专业书 一年时间,如果经济状况没有改善的话,回归企业.

  5. Python 学习笔记(十四)Python类(二)

    创建简单的类 新式类和经典类(旧式类) Python 2.x中默认都是经典类,只有显式继承了object才是新式类 Python 3.x中默认都是新式类,经典类被移除,不必显式的继承object 新式 ...

  6. async函数结合promise的小案例

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  7. node.js 基于cheerio的爬虫工具,需要登录权限的爬虫工具

    公司有过一个需求,需要拿一个网页的的表格数据,数据量达到30w左右:为了提高工作效率. 结合自身经验和网上资料.写了一套符合自己需求的nodejs爬虫工具.也许也会适合你的. 先上代码.在做讲解 'u ...

  8. solr 服务器搭建(Linux版)

    1.下载访问linux工具  securcrt   链接:https://pan.baidu.com/s/1zTwzmnzUmnwbKUCNHCLN1g 密码:kpyg 2.下载所需的 jdk  so ...

  9. 用js实现导出功能将html中的table导出为excel

    /** * 描述:导出表格对应的excel文件 * 时间:2018-03-29 * 作者:任恩远 * 调用示例: * onclick = "tableToExcel(tableId,file ...

  10. 追溯了解Ubuntu之------基本命令操作(叁)

    在使用Ubuntu中的一些基本命令与Linux中是有区别的: 1.       查看Ubuntu系统位数:uname  -ar 或 getconf LONG_BIT 2.          获取Ubu ...