这是数据库开发中经常遇到的问题,当然,这可以用现成的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. Java中用内存映射处理大文件

    在处理大文件时,如果利用普通的FileInputStream 或者FileOutputStream 抑或RandomAccessFile 来进行频繁的读写操作,都将导致进程因频繁读写外存而降低速度.如 ...

  2. Linux- Nginx简单的负载均衡(一)

    这里先进行简单的nginx负载,安装nginx这里就不多说了,我们情景假设在已经安装好了nginx上: 1)查询nginx中的upstrea负载均衡模块  默认是有安装的.进入nginx源码目录中 . ...

  3. python 打印对象的所有属性值

    def prn_obj(obj): print '\n'.join(['%s:%s' % item for item in obj.__dict__.items()])

  4. Linux账号密码过期会导致crontab作业不能执行

    今天一同事报告Linux服务器上的crontab作业没有运行,检查/var/log/cron日志后发现下面错误信息 Jan 19 16:30:01 xxxx crond[31399]: Authent ...

  5. Mongodb Manual阅读笔记:CH2 Mongodb CRUD 操作

    2 Mongodb CRUD 操作 Mongodb Manual阅读笔记:CH2 Mongodb CRUD 操作Mongodb Manual阅读笔记:CH3 数据模型(Data Models)Mong ...

  6. 烂泥:Windows下安装与配置Nginx web服务器

    本文由秀依林枫提供友情赞助,首发于烂泥行天下. 前几篇文章,我们使用nginx都是在linux环境下,今天由于工作的需要.需要在windows环境也使用nginx搭建web服务器. 下面记录下有关ng ...

  7. Unknown class in Interface Builder file 解决方案

    在用swift项目打包Framework时,在项目中使用包时,报错: Unknown class in Interface Builder file... 网上很多解决方案,都不适合我的场景 最终解决 ...

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

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

  9. Python使用QRCode模块生成二维码

    QRCode官网https://pypi.python.org/pypi/qrcode/5.1 简介python-qrcode是个用来生成二维码图片的第三方模块,依赖于 PIL 模块和 qrcode ...

  10. 基于.net开发chrome核心浏览器【五】

    一:本篇将解决的问题 本章主要为了解决一下几个问题: 1.JsDialog的按钮错位的问题 我们开发出的浏览器,在有些操系统上调用alert,confirm之类的对话框时,确定和取消按钮会出现错位的情 ...