TinyFrame升级之七:重构Repository和Unit Of Work
首先,重构的想法来源于以下文章:Correct use of Repository and Unit Of Work patterns in ASP.NET MVC,因为我发现在我的框架中,对UnitOfWork使用了错误的设计方法,同时感谢一下文章:Generically Implementing the Unit of Work & Repository Pattern with Entity Framework in MVC & Simplifying Entity Graphs,我的重构设计参考了它。
下面来说说我的具体重构方式。
在原来的代码中,我的Repository<T>泛型继承自IRepository<T>并且在增删改查代码中做了如下处理:
1: public virtual void Insert(T entity)
2: {
3: try
4: {
5: if (entity == null)
6: throw new ArgumentException("实体类为空");
7: DbSet.Add(entity);
8: context.SaveChanges();
9: }
10: catch (DbEntityValidationException dbex)
11: {
12: var msg = string.Empty;
13: foreach(var validationErrors in dbex.EntityValidationErrors)
14: foreach(var validateionError in validationErrors.ValidationErrors)
15: msg+=string.Format("Property:{0} Error:{1}",validateionError.PropertyName,validateionError.ErrorMessage);
16:
17: var fail = new Exception(msg,dbex);
18: throw fail;
19: }
20: }
最关键的地方是,我添加了“Context.SaveChanges”方法,这就直接导致UnitOfWork的规则失效。UnitOfWork出现的本身是为了提供事务提交支持。这样直接在Repository中提交,直接导致UnitOfWork功能废弃。
还有个地方就是在前台,通过Autofac注册完毕后,我是这么用的:
1: public BookService(IUnitOfWork unitOfWork
2: , IBook bookRepository
3: , IBookType bookTypeRepository
4: , IBookPlace bookPlaceRepository
5: , ICacheManager cacheManager
6: , ILoggerService logger
7: )
8: {
9: this.unitOfWork = unitOfWork;
10: this.bookRepository = bookRepository;
11: this.bookTypeRepository = bookTypeRepository;
12: this.bookPlaceRepository = bookPlaceRepository;
13: this.cacheManager = cacheManager;
14: this.logger = logger;
15: }
16:
17: private readonly IUnitOfWork unitOfWork;
18: private readonly IBook bookRepository;
19: private readonly IBookType bookTypeRepository;
20: private readonly IBookPlace bookPlaceRepository;
21: private readonly ICacheManager cacheManager;
22: private readonly ILoggerService logger;
这样做的话,当以后我们有表删除或者新增的时候,我们不得不维护这样的列表。这完全不符合OO设计原则。
但是如果引入UnitOfWork的话,内部利用Hashtable等实现对Respository的指向,那么在界面我们只要这样写就可以了:
1: public BookService(
2: IUnitOfWork unitOfWork
3: , ICacheManager cacheManager
4: , ILoggerService logger
5: )
6: {
7: this.unitOfWork = unitOfWork;
8: this.cacheManager = cacheManager;
9: this.logger = logger;
10: }
11:
12: private readonly IUnitOfWork unitOfWork;
13: private readonly ICacheManager cacheManager;
14: private readonly ILoggerService logger;
使用的时候,直接这样实例化就行了:
1: var bookPlaceRepository = unitOfWork.Repository<BookPlace>();
这样做完全不用顾虑有新表的添加删除了。代码根本就不用动。
所以,综上两点,UnitOfWork的引入为了解决以下问题:
1.提供全局事务支持。
2.提供对Repository模型的指向。以便于松耦合绑定。
下面是代码重构部分:
IUnitOfWork接口部分:
1: using System;
2: using TinyFrame.Data.DataRepository;
3: using TinyFrame.Data.DomainModel;
4:
5: namespace TinyFrame.Unitofwork
6: {
7: public interface IUnitOfWork
8: {
9: void Commit();
10: IRepository<T> Repository<T>() where T : class;
11:
12: void Dispose(bool disposing);
13: void Dispose();
14: }
15: }
实现部分比较简单,利用Hashtable来保存对Repository的指向:
1: using System;
2: using System.Data.Entity;
3: using TinyFrame.Data.DataRepository;
4: using TinyFrame.Data.DomainModel;
5: using TinyFrame.Data.DataContext;
6: using System.Collections;
7:
8: namespace TinyFrame.Unitofwork
9: {
10: public class UnitOfWork : IUnitOfWork
11: {
12: public UnitOfWork(IDbContext dbContext)
13: {
14: this.dbContext = dbContext;
15: }
16:
17: private readonly IDbContext dbContext;
18: private bool disposed;
19: private Hashtable repositorys;
20:
21: public void Commit()
22: {
23: dbContext.SaveChanges();
24: }
25:
26: public IRepository<T> Repository<T>() where T:class
27: {
28: if (repositorys == null)
29: repositorys = new Hashtable();
30:
31: var type = typeof(T).Name;
32:
33: if (!repositorys.ContainsKey(type))
34: {
35: var repositoryType = typeof(Repository<>);
36: var repositoryInstance = Activator.CreateInstance(repositoryType.MakeGenericType(typeof(T)), dbContext);
37: repositorys.Add(type, repositoryInstance);
38: }
39: return (IRepository<T>)repositorys[type];
40: }
41:
42: #region Dispose method
43: public virtual void Dispose(bool disposing)
44: {
45: if (!disposed)
46: if (disposing)
47: dbContext.Dispose();
48: disposed = true;
49: }
50:
51: public void Dispose()
52: {
53: Dispose(true);
54: GC.SuppressFinalize(this);
55: }
56: #endregion
57: }
58: }
第17行,保持对DbContext的引用,以便于进行提交操作。
第26行,Repository<T>泛型方法,以便于动态返回仓储模型。
需要注意的是,在Repository<T>的实现中,不要再增删改查里面再添加 DbContext.SaveChanges方法,首先是没意义,其次是完全不符合Repository和UnitOfWork的做法。
最后附图一张,表明我对Repository和UnitOfWork的理解:

本章源码下载:
TinyFrame升级之七:重构Repository和Unit Of Work的更多相关文章
- MVC3+EF4.1学习系列(八)-----利用Repository and Unit of Work重构项目
项目最基础的东西已经结束了,但是现在我们的项目还不健全 不利于测试 重复性代码多 层与层之间耦合性高 不利于扩展等问题.今天的这章 主要就是解决这些问题的.再解决这些问题时,自己也产生了很多疑 ...
- 在Entity Framework 4.0中使用 Repository 和 Unit of Work 模式
[原文地址]Using Repository and Unit of Work patterns with Entity Framework 4.0 [原文发表日期] 16 June 09 04:08 ...
- 转载Repository 和Unit of work的使用说明
利用Repository and Unit of Work重构项目 文章索引和简介 项目最基础的东西已经结束了,但是现在我们的项目还不健全 不利于测试 重复性代码多 层与层之间耦合性高 不利于 ...
- TinyFrame升级之一:框架概览
由于之前的TinyFrame多于简单,并且只是说明原理,并无成型的框架出来,所以这次我把之前的知识进行了汇总,然后做出了这一版的TinyFrame框架. 整个框架的结构如下: TinyFrame.Da ...
- Using the Repository and Unit Of Work Pattern in .net core
A typical software application will invariably need to access some kind of data store in order to ca ...
- TinyFrame升级之三:逻辑访问部分
在上一篇,我们打造了自己的数据访问部分,这篇,我们准备讲解如何打造逻辑访问部分. 在上一篇中,我们利用Repository模式构建了基于泛型的操作合集.由于这些操作的合集都是原子性的操作,也就是针对单 ...
- TinyFrame升级之四:IOC容器
在这个框架中,我们使用Autofac作为IOC容器,来实现控制反转,依赖注入的目的. 在程序加载的时候,我需要将系统中所有用到的接口与之对应的实现进行装载.由于用户交互部分是在TinyFrame.We ...
- TinyFrame升级之九:实现复杂的查询
本章我们主要讲解如何实现一个复杂的查询.由于目前TinyFrame框架已经投入到了实际的项目生产中,所以我很乐意将项目中遇到的任何问题做以记录并备忘. 这章中,我们提到的查询界面如下所示: 其中,涉及 ...
- TinyFrame升级之十:WCF Rest Service注入IOC的心
由于在实际开发中,Silverlight需要调用WebService完成数据的获取,由于之前我们一直采用古老的ASMX方式,生成的代理类不仅难以维护,而且自身没有提供APM模式的调用方式,导致在Sin ...
随机推荐
- HTML页面禁止选择、页面禁止复制、页面禁止右键
HTML页面内容禁止选择.复制.右键刚在一个看一个站点的源代码的的时候发现的,其实原来真的很简单 <body leftmargin=0 topmargin=0 oncontextmenu='re ...
- Storm系列(一):搭建dotNet开发Storm拓扑的环境
上篇博客比较了目前流行的计算框架特性,如果你是 Java 开发者,那么根据业务场景选择即可:但是如果你是 .Net 开发者,那么三者都不能拿来即用,至少在这篇文章出现之前是如此.基于上篇文章的比较发现 ...
- Tomcat中取消断点
启动tomcat时,myeclipse报错: This kind of launch is configured to openthe debug perspective when it suspen ...
- Java基础语法总结2
三.运算符 Java基 本 的 运 算 符 按功能分有 下 面 几 类 : 1.算 术 运 算 符 (+,-,*,/,%,++,--) Java对 加 运 算 符 进 行 了 扩 展 ,使 它 能 够 ...
- stm32 加入 USE_STDPERIPH_DRIVER、STM32F10X_HD的原因
初学STM32,在RealView MDK 环境中使用STM32固件库建立工程时,初学者可能会遇到编译不通过的问题.出现如下警告或错误提示: warning: #223-D: function &qu ...
- Mysql 如何实现列值的合并
Mysql 如何实现列值的合并 SELECT GROUP_CONCAT(name SEPARATOR ' ') AS name FROM A
- @RestController注解下返回到jsp视图页面
spring4.1中添加了@RestController注解很方便,集成了@ResponseBody注解,无需再在每个方法前添加了..但是却发现个问题..之前用@Controller注解的时候经常会如 ...
- 报表开发工具中mysql数据库连接编码转化失效解决方案
1. 问题描述 在报表开发工具FineReport中,mysql数据库连接通过数据连接编码转换进行编码的转换,在通过报表录入往数据库中录入中文数据的时候,总是出现乱码,这个该怎么解决呢? 2. 解决方 ...
- (五)适配器模式-C++实现
将一个类的接口转换成客户希望的另外一个接口.Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作. 该模式中有三种角色: 1.目标:是一个抽象类,它是客户想使用的接口 2.被适配 ...
- 在Docker中运行web应用
启动一个简单的web 应用 使用社区提供的模板,启动一个简单的web应用,熟悉下各种Docker命令的使用: # docker run -d -P training/webapp python app ...