跟踪测试 DbContext ,向"不是真正的 ORM" 说拜拜
FreeSql 发展到现在,已经有两种稳定的开发模式,以下先简单带过一下。后面才是本文的主题。
0|1方法一:基于 helper 的方式,祼用;
dotnet add package FreeSql
提供 CodeFirst、DbFirst、丰富的表达式树、读写分离、AOP等功能支持;
0|2方法二:基于 Repository + UnitOfWok 的方式;
dotnet add package FreeSql.Repository
这是一个扩展包,提供标准的 IRepository 接口定义与默认实现,以及 UnitOfWork 工作单元的支持,更可怕的是集成了局部/全局过滤器,实现租户、软删除等功能不在话下。
不相信吗?请看以下代码:
比 abpvnext 还要方便,因为 abp 的相关实体需要实现接口 ISoftDelete、ITenant;
我们没有这个限制,只要过滤器的表达式解析成功,就算可用;
使用在任何实体上的时候,只要 [实体].IsDeleted == false 能解析能过,就算可用;
1|0方式三:基于 DbContext
这个项目仍然是一个扩展包,提类似 EFCore 那样的开发习惯。目前定义的规则如下:
文字规则略显复杂,后边有代码演示,以及图文介绍在 sqlite 和 sqlserver 下的测试过程。
DbContext
- 提供 SaveChanges 方法;
- 执行队列;
DbSet
- 提供 Add、AddRange、Remove、RemoveRange、Update、UpdateRange 方法;
- 以及 Select 属性(连去原有的 FreeSql 查询对象);
- 私有对象 states,存储实体的副本哈希集合,key=实体的主键值,value=实体;
Add/AddRange(entitys)
- 验证 entitys 主键值,是否存在于 states 中,存在时报错;
- 验证 entitys 主键中存在自增:
- 若有,则立即开启 DbContext 事务,按数据库种类执行相应的方法,最终将返回的自增值,赋给entitys的属性;
- 若无,并且 entitys 无主键值,则报错;
- 否则,进入【打包执行队列】;
- 完成时更新 states;
Remove/RemoveRange(entitys)
- 验证 entitys 主键值,若无则报错;
- 验证 states 中是否存在,若无则提醒应该先查询,再删除;
- 删除 states 对应的实体;
- 清除 entitys 内的自增属性值、Guid 类型的值,那这个 entitys 将变为可 Add 状态;
- 进入【打包执行队列】;
Update/UpdateRange(entitys)
- 验证 entitys 主键值,若无则报错;
- 验证 states 中是否存在,若无则提醒应该先查询,再删除;
- 进入【打包执行队列】;
Select
- 立即执行队列中的命令(打包方式),以免脏读到未提交的数据;
- 查询完成时,更新 states 的值;
更新数据规则
- 对比 states 中存在的历史快照值,返回即将修改的 fields;
2|0演示代码
3|0在 sqlite 测试
打个岔:为什么一条条的执行?
- 有自增属性需要获取值;
- sqlite 没有批量插入获取多个自增的办法,或者您有招来支一支(万分感谢);
- 后面采用 sqlserver 测试,就不是这个境况了,insert into values(),(),(),然后利用 output 特性返回所有值;
- 比较蛋疼的是,这个特性不是所有数据库都有
可以看见,最终 SaveChanges 时将不会产生影响的命令,一起打包执行,即采用优化合并的方式进行执行。
例如:
这两个更新操作,会合成一条 SQL 命令执行。
4|0在 sqlserver 测试
其实大致与 sqlite 下相同,唯一的区别在于 AddRange 的处理方式,如图:
当插入单条时,采用了第一行代码的 SQL 命令;
当批量插入时,采用了后面看上去复杂的 SQL 命令;
所有传入的实体属性值在执行完成后,都会更新;
4|1特别说明
FreeSql.DbContext 目前仍处于研究开发阶段,不适合商用;
5|0总结
为什么写这篇文章,时常看见有人说某某 orm 不是真正的 orm,没有 OO 思想。
希望 FreeSql.DbContext 随着时间的积累,稳定性和成熟度有所提升,不久成为一个真正的 ORM。
有人会担心,我们第三方做的不靠谱,没有 EFCore 稳定的说话,这个是当然。
但是我们也有自己的特点,不是吗?我们可以做到多种数据库使用习惯的一致性,这点 EFCore 目前是没有办法解决的难题。
从细节出发,我们的口号是:做 .NETCore 最方便的 ORM!
github: https://github.com/2881099/FreeSql 377星
还请献上宝贵的一星,谢谢观看!!
__EOF__

本文链接:https://www.cnblogs.com/kellynic/p/10580936.html
关于博主:评论和私信会在第一时间回复。或者直接私信我。
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
声援博主:如果您觉得文章对您有帮助,可以点击文章右下角【推荐】一下。您的鼓励是博主的最大动力!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?
· Pantheons:用 TypeScript 打造主流大模型对话的一站式集成库