zoukankan      html  css  js  c++  java
  • Dapper中数据表的字段(列)与实体属性不一致时,如何手动配置它们之间的映射?

    NET[C#]Dapper中数据表的字段(列)与实体属性不一致时,如何手动配置它们之间的映射?

    问题描述

    比如有如下的数据表结构:
    Person:

    person_id  int
    first_name varchar(50)
    last_name  varchar(50)
    

    以及实体类:
    Person:

    public class Person 
    {
        public int PersonId { get; set; }
        public string FirstName { get; set; }
        public string LastName { get; set; }
    }
    

    在C#程序中,使用Dapper做查询时,如何配置数据表字段(列)和实体类属性之间的映射呢?

    方案一

    var sql = @"select top 1 person_id PersonId, first_name FirstName, last_name LastName from Person";
    using (var conn = ConnectionFactory.GetConnection())
    {
        var person = conn.Query<Person>(sql).ToList();
        return person;
    }
    

    方案二

    使用 ColumnAttribute 属性
    完整的代码片段:

    namespace YourNamespace
    {
        /// <summary>
        /// Uses the Name value of the <see cref="ColumnAttribute"/> specified to determine
        /// the association between the name of the column in the query results and the member to
        /// which it will be extracted. If no column mapping is present all members are mapped as
        /// usual.
        /// </summary>
        /// <typeparam name="T">The type of the object that this association between the mapper applies to.</typeparam>
        public class ColumnAttributeTypeMapper<T> : FallbackTypeMapper
        {
            public ColumnAttributeTypeMapper()
                : base(new SqlMapper.ITypeMap[]
                    {
                        new CustomPropertyTypeMap(
                           typeof(T),
                           (type, columnName) =>
                               type.GetProperties().FirstOrDefault(prop =>
                                   prop.GetCustomAttributes(false)
                                       .OfType<ColumnAttribute>()
                                       .Any(attr => attr.Name == columnName)
                                   )
                           ),
                        new DefaultTypeMap(typeof(T))
                    })
            {
            }
        }
    
            [AttributeUsage(AttributeTargets.Property, AllowMultiple = true)]
        public class ColumnAttribute : Attribute
        {
            public string Name { get; set; }
        }
    
          public class FallbackTypeMapper : SqlMapper.ITypeMap
        {
            private readonly IEnumerable<SqlMapper.ITypeMap> _mappers;
    
            public FallbackTypeMapper(IEnumerable<SqlMapper.ITypeMap> mappers)
            {
                _mappers = mappers;
            }
    
    
            public ConstructorInfo FindConstructor(string[] names, Type[] types)
            {
                foreach (var mapper in _mappers)
                {
                    try
                    {
                        ConstructorInfo result = mapper.FindConstructor(names, types);
                        if (result != null)
                        {
                            return result;
                        }
                    }
                    catch (NotImplementedException)
                    {
                    }
                }
                return null;
            }
    
            public SqlMapper.IMemberMap GetConstructorParameter(ConstructorInfo constructor, string columnName)
            {
                foreach (var mapper in _mappers)
                {
                    try
                    {
                        var result = mapper.GetConstructorParameter(constructor, columnName);
                        if (result != null)
                        {
                            return result;
                        }
                    }
                    catch (NotImplementedException)
                    {
                    }
                }
                return null;
            }
    
            public SqlMapper.IMemberMap GetMember(string columnName)
            {
                foreach (var mapper in _mappers)
                {
                    try
                    {
                        var result = mapper.GetMember(columnName);
                        if (result != null)
                        {
                            return result;
                        }
                    }
                    catch (NotImplementedException)
                    {
                    }
                }
                return null;
            }
    
    
            public ConstructorInfo FindExplicitConstructor()
            {
                return _mappers
                    .Select(mapper => mapper.FindExplicitConstructor())
                    .FirstOrDefault(result => result != null);
            }
        }
    
    }
    

    调用方法:

    public class Person 
    {
         [Column(Name="person_id")]
        public int PersonId { get; set; }
        [Column(Name="first_name")]
        public string FirstName { get; set; }
        [Column(Name="last_name")]
        public string LastName { get; set; }
    }
    

    方案三

    使用 CustomPropertyTypeMap 自定义属性类型映射类,如:

    public class ColumnMap
    {
        private readonly Dictionary<string, string> forward = new Dictionary<string, string>();
        private readonly Dictionary<string, string> reverse = new Dictionary<string, string>();
    
        public void Add(string t1, string t2)
        {
            forward.Add(t1, t2);
            reverse.Add(t2, t1);
        }
    
        public string this[string index]
        {
            get
            {
                // Check for a custom column map.
                if (forward.ContainsKey(index))
                    return forward[index];
                if (reverse.ContainsKey(index))
                    return reverse[index];
    
                // If no custom mapping exists, return the value passed in.
                return index;
            }
        }
    }
    

    配置列映射关系

    var columnMap = new ColumnMap();
    columnMap.Add("Field1", "Column1");
    columnMap.Add("Field2", "Column2");
    columnMap.Add("Field3", "Column3");
    
    SqlMapper.SetTypeMap(typeof (MyClass), new CustomPropertyTypeMap(typeof (MyClass), (type, columnName) => type.GetProperty(columnMap[columnName])));
    

    方案四

    查询时使用LINQ

     var sql = @"select top 1 person_id, first_name, last_name from Person";
        using (var conn = ConnectionFactory.GetConnection())
        {
            List<Person> person = conn.Query<dynamic>(sql)
                                      .Select(item => new Person()
                                      {
                                          PersonId = item.person_id,
                                          FirstName = item.first_name,
                                          LastName = item.last_name
                                      }
                                      .ToList();
    
            return person;
        }
    

    http://2sharings.com/2018/dapper-orm-manually-map-column-names-with-class-properties-in-csharp-application
  • 相关阅读:
    搜索文字高亮
    聊天 页面 定位overflow: scroll 原生滚动,iOS下会不流畅,
    MySQL技术专题(X)该换换你的数据库版本了,让我们一同迎接8.0的到来哦!(初探篇)
    Zookeeper原理系列-Paxos协议的原理和Zookeeper中的应用分析
    【Spring技术原理】Aspectj和LoadTimeWeaving的动态代理技术实现指南
    【SpringBoot技术专题】「权限校验专区」Shiro整合JWT授权和认证实现
    👊 Spring技术原理系列-从零开始教你SpringEL表达式使用和功能分析讲解指南(上篇)
    SpringBoot-技术专区-教你如何开发一个”可移植“的轻量级文件服务项目系统!
    ☕【Java技术指南】「JPA编程专题」让你不再对JPA技术中的“持久化型注解”感到陌生了!
    【SpringCloud技术专题】「Eureka源码分析」从源码层面让你认识Eureka工作流程和运作机制(下)
  • 原文地址:https://www.cnblogs.com/efreer/p/8277329.html
Copyright © 2011-2022 走看看