这是数据库开发中经常遇到的问题,当然,这可以用现成的ORM框架来解决,但有些时候,如果DataSet/DataTable是第三方接口返回的,ORM就不方便了,还得自己处理。

反射自然必不可少的,另外考虑到DataTable中的ColumnName通常与Model的PropertyName并不严格对应,可以用Attribute来记录这种映射关系。

步骤1:先创建一个DataFieldAttribute类

 using System;

 namespace Jimmy.ORM
{
[AttributeUsage(AttributeTargets.Property)]
public sealed class DataFieldAttribute:Attribute
{
/// <summary>
/// 表对应的字段名
/// </summary>
public string ColumnName { set; get; } public DataFieldAttribute(string columnName)
{
ColumnName = columnName;
}
}
}

步骤2:在Model/Entity的Class成员上,应用DataField特性,参见下面的代码:

 using System;

 namespace Jimmy.ORM.Entity
{
[Serializable]
public class ProductEntity : DataEntityBase
{ [DataField("PRODUCT_NO")]
public string ProductNo { set; get; } [DataField("PRODUCT_ID")]
public int ProductId { set; get; } [DataField("PRODUCT_NAME")]
public string ProductName { set; get; } public override string ToString()
{
return string.Format("ProductNo:{1}{0}ProductId:{2}{0}ProductName:{3}", Environment.NewLine, ProductNo,
ProductId, ProductName);
}
}
}

步骤3:该反射出场了,为了方便起见,封装了一个DataConvert类

 using System;
using System.Collections.Generic;
using System.Data;
using System.Reflection; namespace Jimmy.ORM
{
/// <summary>
/// 将DataRow/DataTable转换成Entity/Entity列表
/// </summary>
public static class DataConvert<T> where T : DataEntityBase, new()
{
/// <summary>
/// 将DataRow行转换成Entity
/// </summary>
/// <param name="dr"></param>
/// <returns></returns>
public static T ToEntity(DataRow dr)
{
T entity = new T();
Type info = typeof(T);
var members = info.GetMembers();
foreach (var mi in members)
{
if (mi.MemberType == MemberTypes.Property)
{
//读取属性上的DataField特性
object[] attributes = mi.GetCustomAttributes(typeof(DataFieldAttribute), true);
foreach (var attr in attributes)
{
var dataFieldAttr = attr as DataFieldAttribute;
if (dataFieldAttr != null)
{
var propInfo = info.GetProperty(mi.Name);
if (dr.Table.Columns.Contains(dataFieldAttr.ColumnName))
{
//根据ColumnName,将dr中的相对字段赋值给Entity属性
propInfo.SetValue(entity,
Convert.ChangeType(dr[dataFieldAttr.ColumnName], propInfo.PropertyType),
null);
} }
}
}
}
return entity;
} /// <summary>
/// 将DataTable转换成Entity列表
/// </summary>
/// <param name="dt"></param>
/// <returns></returns>
public static List<T> ToList(DataTable dt)
{
List<T> list = new List<T>(dt.Rows.Count);
foreach (DataRow dr in dt.Rows)
{
list.Add(ToEntity(dr));
}
return list;
}
}
}

步骤4:测试

 using System;
using System.Data;
using Jimmy.ORM.Entity; namespace Jimmy.ORM.Test
{
class Program
{
static void Main()
{
DataTable dt = new DataTable();
dt.Columns.Add("PRODUCT_NO");
dt.Columns.Add("PRODUCT_ID");
dt.Columns.Add("PRODUCT_NAME"); dt.Rows.Add("", , "手机");
dt.Rows.Add("", , "服装"); var products = DataConvert<ProductEntity>.ToList(dt); foreach (var entity in products)
{
Console.WriteLine(entity);
} Console.Read();
}
}
}

示例代码下载:http://files.cnblogs.com/yjmyzz/DataConvertor.zip

C#:DataTable映射成Model的更多相关文章

  1. C# DataTable映射成Entity

    using System; using System.Collections.Generic; using System.ComponentModel.DataAnnotations.Schema; ...

  2. DataTable转化为Model

    /// <summary> /// 将DataTable转成Model /// </summary> /// <param name="dt"> ...

  3. datatable转换为list<model> 映射

    using System; using System.Collections.Generic; using System.Data; using System.Linq; using System.R ...

  4. DataTable转换成List<T>

    很多时候需要将DataTable转换成一组model,直接对model执行操作会更加方便直观. 代码如下: public static class DataTableToModel { public ...

  5. C# 中DataTable转成模型List

    C# 中DataTable转成模型List 引入using System.Reflection; 命名空间 使用注意实体类的属性名必须和DataTable的列名一致 使用: DBList<Sto ...

  6. DbUtility-关于DataTable转成List的效率问题

    DbUtility中的方法ExecuteDataTableAsync()得到的是一个DataTable,而我们常见的情况下,我们需要的不是DataTable,而是List或IList,所以现在需要考虑 ...

  7. C# DataTable转换成实体列表 与 实体列表转换成DataTable

    /// <summary> /// DataTable转换成实体列表 /// </summary> /// <typeparam name="T"&g ...

  8. C# 中 DataTable转换成IList

    在用C#作开发的时候经常要把DataTable转换成IList:操作DataTable比较麻烦,把DataTable转换成IList,以对象实体作为IList的元素,操作起来就非常方便. 注意:实体的 ...

  9. DataTable转换成IList

    //文章出处: http://www.cnblogs.com/hlxs/archive/2011/05/09/2087976.html DataTable转换成IList 在用C#作开发的时候经常要把 ...

随机推荐

  1. Computer Network and Internet(1)

    计算机网路相关的教材很少,TCP/IP,HTTP 协议非常多,很难找到一个合适的材料去学习. <计算机网络>自上而下方法是这个方面的经典之作. 1.what is internet? 1. ...

  2. 在ROS中使用Python3

    Use Python3 in ROS. 以下内容在Ubuntu 16.04 x64和ROS kinetic中测试通过 事实上,只要在.py文件加上python3的shebang,rosrun的时候就会 ...

  3. Maven项目无法引入 Maven Dependencies Libraries 问题

    昨天在check下来maven项目之后一些配置好了,就是下载不是maven 依赖库,后面再网上找到如下解决方案. 在.classpath文件中加入如下代码就好了. <classpathentry ...

  4. SQL Server(六)——索引、视图和SQL编程

    1.索引 添加索引,设计界面,在任何一列前右键--索引/键--点击进入添加某一列为索引 2.视图 视图就是我们查询出来的虚拟表 创建视图:create view 视图名 as SQL查询语句,分组,排 ...

  5. C#显示SQL语句格式

    --SQL SERVER生成测试环境: Create database Test; go USE [Test] GO if OBJECT_ID('Tab','U') is not null drop ...

  6. MySQL创建索引语法

    1.介绍: 所有mysql索引列类型都可以被索引,对来相关类使用索引可以提高select查询性能,根据mysql索引数,可以是最大索引与最小索引,每种存储引擎对每个表的至少支持16的索引.总索引长度为 ...

  7. 讲讲js中的逻辑与(&&)以及逻辑或(||)

    前几天看到一个函数,百思不得其解,今天早上醒来看了本js的书,正好讲到操作符的用法,给大家分享下js中的&&,||,和我们用的其他的编程语言还是有点区别的. 直接上那个函数的代码: f ...

  8. Windows Sever关于80端口之争

    默认情况下安装了IIS服务器角色的Windows系统,其80端口就被占用了.但是占用80端口的进程却不是WWW Service更不是IIS Admin Sevice,而是处于kernel地位的Http ...

  9. php cli方式下获取服务器ip

    (未整理....) (1)php cli方式下获取服务器ip [php]  function getServerIp(){          $ss = exec('/sbin/ifconfig et ...

  10. Android中使用dimen定义尺寸(转)

    (转自:http://blog.csdn.net/yuzhiboyi/article/details/7696174) 最近,遇到了一种定义尺寸的方法,类似于C里面的宏定义一样,其实以前已经见过了这种 ...