一个ORM的实现(附源代码)
1 前言
经过一段时间的编写,终于有出来一个稳定的版本,期间考虑了多种解决方案也偷偷学了下园子里面大神们的作品。
已经有很多的ORM框架,为什么要自己实现一个?我的原因是在遇到特殊需求时,可以在ORM中加入特定的代码。如 :根据数据库的字段长度和可空性做基本的数据验证,在ORM中解决数据修改时的同步问题…
本文主要关注的是如何实现ORM方面,其它的大家可以参考以下两篇文章:
用T4 Template生成代码:参考此文,可以知道本ORM是如何根据数据库,生成实体层代码。有了这个基础,就可以看懂本文本中所有的T4模板。
DBHelper (支持事务与数据库变更):参考此文,可以知道本ORM是如何访问数据库,如何支持事务,分页和多种数据库。
2 实现方法
本文主要关注的蓝色ORM部分。
Model:定义了实体的基类。每一张表都会生成对应的实体并继承此类;
ModelMapping:定义了一张表的结构信息,其中包含了表名、主键、字段类型等。每一张表都会生成一个;
ModelMappingReflector: 在程序第一次运行时,反射所有ModelMapping的子类,并将其加入一个字典集合中;
DataAccess<T>: 此类的作用时,根据实体和实体的结构信息(ModelMapping),自动生成SQL来访问数据库(Add,Update,Delete);或将数据库中值加到实体上(GetModel,GetList)。此类帮我们做CRUD的工作,我们不再需要写每张表基本的增删改查的工作了,但仍需要写复杂业务的查询。每一张表都会生成一个空的dao类并继承此类。
此ORM的核心类就是DataAccess<T>, 下面我们就拿Add作为示例:
Add 操作代码
public int Add(T model)
{
if (model == null)
{
throw new ArgumentNullException();
} StringBuilder cmdText = new StringBuilder();
cmdText.Append("INSERT INTO ").Append(modelMapping.TableName).Append(" (");
cmdText.Append(string.Join(",", model.ColumnValues.Keys.ToArray()));
cmdText.Append(") VALUES (@");
cmdText.Append(string.Join(", @", model.ColumnValues.Keys.ToArray()));
cmdText.Append(")"); List<DbParameter> dbParms = new List<DbParameter>();
foreach (var pair in model.ColumnValues)
{
dbParms.Add(DBHelper.CreateInDbParameter(string.Format("@{0}",pair.Key), modelMapping.ColumnDict[pair.Key].DbDataBype, pair.Value));
} return this.ExecuteNonQuery(CommandType.Text, cmdText.ToString(), dbParms.ToArray());
}
当我们写这样的代码时User model=new User(); model.UserName时,实际上会向Model中ColumnValues集合中添加数据。而要生成一个INSERT SQL (INSERT INTO tablename (column1,colum2) VALUES (@column1,@column2)和添加插入参数,我们需要表名,列名,列值和列的类型。在列名和列值就存在于Model中的ColumnValues集合,而列的类型和表名,我们可以在ModelMapping中获取。
3 示例
static void Main(string[] args)
{
TestORM();
TestPaging(); Console.ReadLine();
} static void TestORM()
{
UserDao dao = new UserDao();
try
{
dao.BeginTransaction();
// add model
User mUser1 = new User();
mUser1.ID = "";
mUser1.UserName = "Mike1";
mUser1.UserPwd = "";
mUser1.CreateBy = "System";
mUser1.CreateDate = DateTime.Now;
dao.Add(mUser1); Console.WriteLine("Create Model Successfully.");
string cmdText = "SELECT * FROM TUser WHERE ID='1' ORDER BY ID;";
List<User> list0 = dao.GetListBySQL(cmdText);
foreach (User model in list0)
{
Console.WriteLine(model.UserName);
}
Console.WriteLine("Get List Successfully."); //嵌套事务
UserDao dao2 = new UserDao();
try
{
dao2.BeginTransaction();
//update model
User mUser2 = new User();
mUser2.ID = "";
mUser2.UserName = "Mike.Jiang";
mUser2.UserPwd = null;
mUser2.CreateDate = null;
dao.Update(mUser2);
Console.WriteLine("Update Model Successfully."); User mUser3 = dao.GetModel("");
if (mUser3 != null)
Console.WriteLine(mUser3.UserName); Console.WriteLine("Get Model Successfully.");
dao2.CommitTransaction();
}
catch
{
dao2.RollbackTransaction();
} dao.Delete(""); Console.WriteLine("Delete Model Successfully.");
dao.CommitTransaction();
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
dao.RollbackTransaction();
}
} static void TestPaging()
{
UserDao dao = new UserDao();
// add 100 user
for (int i = ; i <= ; i++)
{
User mUser1 = new User();
mUser1.ID = i.ToString();
mUser1.UserName = "Mike" + i.ToString();
mUser1.UserPwd = "";
mUser1.CreateBy = "System";
mUser1.CreateDate = DateTime.Now;
dao.Add(mUser1);
} string cmdText = string.Format("SELECT * FROM TUser");
DataTable table = DBHelper.ExecutePagingDataTable(CommandType.Text, cmdText, , , "ORDER BY ID"); foreach (DataRow row in table.Rows)
{
Console.WriteLine(row["ID"]);
} Console.WriteLine("Get Datatabe paging successfuly."); List<User> list = dao.GetListPaging(cmdText, , , "ORDER BY ID");
foreach (User m in list)
{
Console.WriteLine(m.ID);
} Console.WriteLine("Get Model list paging successfully."); cmdText = "DELETE FROM TUser";
dao.ExecuteNonQuery(CommandType.Text, cmdText);
}
4 总结
这个版本的ORM示例,示例是没有认真去写了,因为不认为会有人用。但是这个ORM的代码是我认为比较简单的一个版本,只具有ORM的核心功能,几个类,只要具有c#基础的人都能看懂,有兴趣的可以看下。后续,自己会加上分页的Pager相关的内容、数据同步、数据验证的功能和和根据SQL自动生成实体对象的功能。
所有的源代码: BaseProject.7z
一个ORM的实现(附源代码)的更多相关文章
- 效率最高的Excel数据导入---(c#调用SSIS Package将数据库数据导入到Excel文件中【附源代码下载】) 转
效率最高的Excel数据导入---(c#调用SSIS Package将数据库数据导入到Excel文件中[附源代码下载]) 本文目录: (一)背景 (二)数据库数据导入到Excel的方法比较 ...
- 【转】P2P之UDP穿透NAT的原理与实现(附源代码)
作者:shootingstars (有容乃大,无欲则刚) 日期:2004-5-25 出处:P2P中国(PPcn.net) P2P 之 UDP穿透NAT的原理与实现(附源代码)原创:shootings ...
- 仿36氪(iOS版附源代码)
前言: 这是我2016年3月开始写的,利用课余时间全心投入的项目,本以为是凭着轻松愉悦的方式来学习的,中途遇到bug解决bug的时候,每天晚上几乎都是写到寝室关灯,还有一次使用Github不当写了五天 ...
- WCF技术剖析之二十八:自己动手获取元数据[附源代码下载]
原文:WCF技术剖析之二十八:自己动手获取元数据[附源代码下载] 元数据的发布方式决定了元数据的获取行为,WCF服务元数据架构体系通过ServiceMetadataBehavior实现了基于WS-ME ...
- Java设计模式-代理模式之动态代理(附源代码分析)
Java设计模式-代理模式之动态代理(附源代码分析) 动态代理概念及类图 上一篇中介绍了静态代理,动态代理跟静态代理一个最大的差别就是:动态代理是在执行时刻动态的创建出代理类及其对象. 上篇中的静态代 ...
- Spring Web Flow 入门demo(三)嵌套流程与业务结合 附源代码
上篇博客我们说Spring web Flow与业务结合的方式主要有三种,以下我们主要介绍一下第三种的应用方式 3,运行到<action-state> 元素 SpringWeb Flow 中 ...
- 基于事件驱动的DDD领域驱动设计框架分享(附源代码)
原文:基于事件驱动的DDD领域驱动设计框架分享(附源代码) 补充:现在再回过头来看这篇文章,感觉当初自己偏激了,呵呵.不过没有以前的我,怎么会有现在的我和现在的enode框架呢?发现自己进步了真好! ...
- 手把手实现微信网页授权和微信支付,附源代码(VUE and thinkPHP)
wechat github 手把手实现微信网页授权和微信支付,附源代码(VUE and thinkPHP) 概述 公众号开发是痛苦的,痛苦在好多问题开发者文档是没有提到的,是需要你猜的. 在开发过程中 ...
- 介介介是一个ORM
介个是一个ORM,介个ORM基于Dapper扩展. 为什么需要一个ORM呢? 支持简单的LINQ查询 但是不能连表查询,why?why?why?为什么不能连接查询 ^.^ ok.但是就是不支持.哈哈哈 ...
- java程序保护如何知识产权,特别提供一个java 开发的java 源代码级的混淆器
java程序保护如何知识产权,特别提供一个java 开发的java 源代码级的混淆器 下载地址:http://yunpan.cn/QXhEcGNYLgwTD 运行方式:java -jar Encryp ...
随机推荐
- 【.net】从比较两个字节数组谈起
上午,有位初学者朋友问:如何比较两个字节数组中各字节是否相等? 不许笑,我一向反对嘲笑初学者,初学者不认真学习时你可以批评,但不能讥嘲.你不妨想想,你自己开始学习编程的时候又是什么个光景? 好,于是, ...
- Java多线程系列--“基础篇”06之 线程让步
概要 本章,会对Thread中的线程让步方法yield()进行介绍.涉及到的内容包括:1. yield()介绍2. yield()示例3. yield() 与 wait()的比较 转载请注明出处:ht ...
- 默認打開pr_debug和dev_dbg
作者:彭東林 郵箱:pengdonglin137@163.com 日期:2016-08-26 18:04:14 在進行Linux驅動開發時經常見到使用pr_debug和dev_dbg打印驅動的log, ...
- Android基于mAppWidget实现手绘地图(八)–获取用户地理位置
你有两种方式确定用户的当前位置: 1.使用安卓标准方式 2. 使用MapWidget的监听器 标准方式: 在你的acivity中加入: LocationManager locManager = (Lo ...
- Android中动态更新ListView(转)
在使用ListView时,会遇到当ListView列表滑动到最底端时,添加新的列表项的问题,本文通过代码演示如何动态的添加新的列表项到ListView中.实现步骤:调用ListView的setOnSc ...
- equals和==的区别 你真的掌握了吗?
PS:最近读Java编程思想的时候发现了一些小问题.就是equals方法和==,感觉自己是真正掌握了,其实并没有.简单的记录一下. 学习内容: 1.equals 和 == 的区别 equals和== ...
- Windows Azure HandBook (4) 分析Windows Azure如何处理Session
<Windows Azure Platform 系列文章目录> 本文是对笔者之前的文章Windows Azure Cloud Service (13) 多个VM Instance场景下如何 ...
- C# Socket系列三 socket通信的封包和拆包
通过系列二 我们已经实现了socket的简单通信 接下来我们测试一下,在时间应用的场景下,我们会快速且大量的传输数据的情况! class Program { static void Main(stri ...
- 用Qt写软件系列五:一个安全防护软件的制作(2)
引言 在上一篇中讲述了主窗体的创建和设计.主窗体的无边框效果.阴影效果.拖动事件处理.窗体美化等工作在前面的博客中早就涉及,因此上篇博文中并未花费过多笔墨.这一篇继续讲述工具箱(Tool Button ...
- [Offer收割]编程练习赛3 - 题目3 : 智力竞赛
智力竞赛 Problem's Link ---------------------------------------------------------------------------- Mea ...