C# 数据操作系列 - 4. 自己实现一个ORM
0. 前言
在之前的几篇内容中,我们了解了如何通过ADO.NET 访问数据库,如何修改、新增数据。如何通过DataSet和DataAdapter获取数据,我们将在这一篇试试自己实现一个简单的ORM框架或者说ORM工具类。
涉及到的知识点:
- 反射(初级)
- ADO.NET 已有知识
1. ORM
那么,问题来了,什么是ORM?ORM全称 Object Relational Mapping,翻译过来就是对象关系映射。是一种通过描述对象与数据库之间映射关系的数据,将对象保存到数据库中的技术。
在C#中,曾经Entity Framework光芒万丈,遮盖了其他ORM框架的光辉(甚至如今都是如此)。
后来慢慢涌现除了其他的一些ORM框架,进一步丰富了市场。所以现有比较流行的大概有以下几种:
- Dapper 一个轻量的ORM框架
- Entity Framework/Entity Framework Core 功能完备的框架
- Nhibernate Java平台上著名的Hibernate的.net版
- 等等
嗯,这是我最近找到的创作组还在更新的几个框架,当然还有其他的很多有趣好用的ORM框架。欢迎各位补充哈。
这一篇的主要目的不是介绍这些框架(这是以后的内容),而是通过我们自己实现一个类ORM框架来了解底层核心。
2. 设计
我们先分析一下,如果我们设计一个实体对象与数据库之间转换的工具类应该具有哪些功能?
- 一个属性与数据库字段的映射关系
- 增删改查的SQL模板
- 查询结果与对象的转换
3. 实现
首先,声明一个类,因为不能仅支持一种类型,所以这个类的所有与数据库有关的方法都是泛型方法,或者这个类是泛型类,所以定义为泛型类:
public class OrmUtil<T>
{
}
我们事先约定类名即表名,属性名即表的列名,所以我们可以快速得到以下内容:
/// <summary>
/// T的类型实例
/// </summary>
private Type dType;
/// <summary>
/// T的属性表
/// </summary>
private PropertyInfo[] properties;
public OrmUtil()
{
dType = typeof(T);
properties = dType.GetProperties();
}
声明一个数据库连接:
public SqlConnection Connection { get; set; }
创建一个私有方法,检查连接是否可用:
/// <summary>
/// 检查连接是否可用
/// </summary>
/// <returns></returns>
private bool CheckConnection()
{
return Connection?.State == ConnectionState.Open;
}
准备工作完成,然后开始编写具体的业务方法:
Insert:
public int Insert(T entity)
{
if (!CheckConnection()) return -1;// 检查状态
var insert = $"insert into {dType.Name}({string.Join(",", properties.Select(t => t.Name))})";
var values = properties.Select(p => p.GetValue(entity));
var commandText = $"{insert} values('{string.Join("','", values)}')";
var command = Connection.CreateCommand();
command.CommandText = commandText;
var result = command.ExecuteNonQuery();
return result;
}
首先按照属性名与列名之间的映射拼接 SQL,然后执行SQL命令。
Update:
public int Update(T entity,string keyName,object keyValue)
{
if (!CheckConnection()) return -1;
var setValues = properties.ToDictionary(p => p.Name, p => $"'{p.GetValue(entity)}'");
var setSql = string.Join(",", setValues.Select(pair=>$"{pair.Key}='{pair.Value}'"));
var sql = $"update {dType.Name} set {setSql} where {keyName} = '{keyValue}'";
var command = Connection.CreateCommand();
command.CommandText = sql;
return command.ExecuteNonQuery();
}
Update需要注意的就是如何正确拼接赋值sql。
Delete:
删除满足条件的对象:
public int Delete(T entity)
{
if (!CheckConnection()) return -1;
var querySet = properties.Select(p => $"{p.Name} = '{p.GetValue(entity)}'");
var sql = $"delete from {dType.Name} where {string.Join(" and ", querySet)}";
var command = Connection.CreateCommand();
command.CommandText = sql;
return command.ExecuteNonQuery();
}
这里写法有时候根据实际业务不同,大多数情况下删除主键对应的元素,或者满足某一个条件的所有元素。这里只是做了个演示,小伙伴们可以试试自己改造一下。
Search:
先创建一个从DataTable转成对象的工具方法:
private List<T> Convert(DataTable table)
{
var list = new List<T>(table.Rows.Count);//事先声明一下容量
foreach(DataRow row in table.AsEnumerable())
{
T entity = Activator.CreateInstance<T>();
foreach(var p in properties)
{
if (!table.Columns.Contains(p.Name)) continue;// 如果属性名不在表格中,则忽略
p.SetValue(entity, row[p.Name]);
}
list.Add(entity);
}
return list;
}
好,我们写一个查询方法:
public List<T> SearchAll()
{
var adapter = new SqlDataAdapter($"select * from {dType.Name}", Connection);
var set = new DataSet();
adapter.Fill(set);
return Convert(set.Tables[0]);
}
这样一个简单的ORM框架就这样形成雏形了,当然实际上的ORM底层比这复杂,因为需要支持不同的数据库,所以Connection 就不能简简单单的是一个SqlConnection了,或者底层不是像我们一样取巧使用DataTable了。
实际上的DataTable到类对象的转换要比我写的复杂一点,因为还要判断这个属性是否是可读、可写的。
4. 总结
在这里我做了个抛砖引玉,带领小伙伴们一起构思了一个简陋的ORM框架,也让大伙对此有了一定的印象。嗯,今天就到这了。同时ADO.NET 也告一段落了,接下来就是上Entity Framework了。当然,DataSet、DataAdapter这两个类并没有讲完。这部分内容可能会在后续的番外篇内补全。
更多内容烦请关注我的博客《高先生小屋》

C# 数据操作系列 - 4. 自己实现一个ORM的更多相关文章
- C# 数据操作系列 - 6 EF Core 配置映射关系
0. 前言 在<C# 数据操作系列 - 5. EF Core 入门>篇中,我们简单的通过两个类演示了一下EF增删改查等功能.细心的小伙伴可能看了生成的DDL SQL 语句,在里面发现了些端 ...
- C# 数据操作系列 - 8. EF Core的增删改查
0.前言 到目前为止,我们看了一下如何声明EF Core的初步使用,也整体的看了下EF Core的映射关系配置以及导航属性的配置. 这一篇,我带大家分享一下,我在工作中需要的EF Core的用法. 1 ...
- C# 数据操作系列 - 12 NHibernate的增删改查
0. 前言 上一篇<C# 数据操作系列 - 11 NHibernate 配置和结构介绍> 介绍了Nhibernate里的配置内容.这一篇将带领大家了解一下如何使用NHIbernate.之前 ...
- C# 数据操作系列 - 15 SqlSugar 增删改查详解
0. 前言 继上一篇,以及上上篇,我们对SqlSugar有了一个大概的认识,但是这并不完美,因为那些都是理论知识,无法描述我们工程开发中实际情况.而这一篇,将带领小伙伴们一起试着写一个能在工程中使用的 ...
- C# 数据操作系列 - 16 SqlSugar 完结篇
0. 前言 前一篇我们详细的介绍了SqlSugar的增删改查,那些已经满足我们在日常工程开发中的使用了.但是还有一点点在开发中并不常用,但是却非常有用的方法.接下来让我们一起来看看还有哪些有意思的内容 ...
- C# 数据操作系列 - 19 FreeSql 入坑介绍
0. 前言 前几天FreeSql的作者向我推荐了FreeSql框架,想让我帮忙写个文章介绍一下.嗯,想不到我也能带个货了.哈哈,开个玩笑-看了下觉得设计的挺有意思的,所以就谢了这篇文章. 简单介绍一下 ...
- C# 数据操作系列 - 1. SQL基础操作
0.前言 前篇介绍了一些数据库的基本概念和以及一些常见的数据库,让我们对数据库有了一个初步的认识.这一篇我们将继续为C#数据操作的基础填上一个空白-SQL语句. SQL(Structured Quer ...
- C# 数据操作系列 - 0. 序言
0. 前言 在上一个系列中,我们初步浏览了一下C#的基础知识.这句话的意思就是C#基础知识系列完结了,撒花.当然,并不是因为C#已经讲完了.正是因为我们轻轻地叩开了那扇门,才能看到门后面那瑰丽的世界. ...
- C# 数据操作系列 - 2. ADO.NET操作
0.前言 在上一篇中初略的介绍了一下SQL的基本写法,这一篇开始我们正式步入C#操作数据库的范围.通过这一系列的内容,我想大家能对于数据库交互有了一定的认识和基础.闲话不多说,先给大家介绍一个C#操作 ...
随机推荐
- ubuntu允许root远程登录
之前说了,怎么设置root用户登录,但是使用Xshell等工具操作Ubuntu,用户登录不进去,原因很简单,没有开启root登录的权限. 1.检查ssh服务是否开启 使用以下命令,查看是否开启 ps ...
- adb命令查看手机应用内存使用情况
adb shell回车 一.procrank VSS >= RSS >= PSS >= USSVSS - Virtual Set Size 虚拟耗用内存(包含共享库占用的内存)是单个 ...
- 今天我们来讨论一下CSS3属性中的transition属性;
transition属性是CSS3属性:顾名思义英文为过渡的意思:主要有四个值与其一一对应:分别是property(CSS属性名称),duration过渡的时长,timimg-function转速曲线 ...
- spring boot连接linux服务器上的redis
本文章为给新手学习spring boot远程连通redis提供一个学习参考. 环境是intellij idea(window)+ redis(linux虚拟机-vmware). 首先在linux安装好 ...
- 进程管理工具 Supervisor
要想在终端后台常驻进程,首先想到的是在命令后加 & 符号,来达到隐藏程序在后台的目的,尽管看起来进程已经在后台运行了,实际上终端会话关闭时进程还是会被 kill 掉,这种问题一般是采用搭配 n ...
- 大部分人都不知道的8个python神操作
01 print 打印带有颜色的信息 大家知道 Python 中的信息打印函数 Print,一般我们会使用它打印一些东西,作为一个简单调试. 但是你知道么,这个 Print 打印出来的字体颜色是可以设 ...
- 算法笔记刷题1(codeup 1934)
准备6月份的拼题甲级中(本来现在这两天就考试了,但是因为疫情的原因延期了) 刚刚开始按算法笔记刷题,今天是探索codeup的第一天. 一开始并没有把多点测试当回事,直到一错再错,心态爆炸... 附上我 ...
- STM32 内存分配解析及变量的存储位置
内存映射 在一些桌面程序中,整个内存映射是通过虚拟内存来进行管理的,使用一种称为内存管理单元(MMU)的硬件结构来将程序的内存映射到物理RAM.在对于 RAM 紧缺的嵌入式系统中,是缺少 MMU 内存 ...
- 7、窗口函数 & Windows 的 Operator demo
代码地址:https://gitee.com/nltxwz_xxd/abc_bigdata 一.窗口函数 在定义了窗口分配器之后,我们需要为每一个窗口明确的指定计算逻辑,这个就是窗口函数要做的事情,当 ...
- MySQL事务与并发
很多程序员都学过MySQL,而且也会写SQL语句.但仅仅会写还远远不够,在面试中以及在工作中,还必须要会事务和并发. 一.事务 事务是满足 ACID 特性的操作,可以通过 Commit 提交事务, ...