使用PetaPoco结合注入实现业务级事务
使用PetaPoco结合注入实现业务级事务
PetaPoco是一个轻量级ORM,我的MVC项目中使用它结合Repository模式,依靠Unity的生命周期管理对象,保证请求/线程级别的数据上下文单例,并使用锁和计数实现业务级事务。下文代码依个人理解实现,谬误请不吝指正。
例行IUnitOfWork:
public interface IUnitOfWork
{
void Begin();
void Commit();
void Rollback();
}
仓库上下文核心:

1 public class PetaPocoUnitOfWork : IUnitOfWork
2 {
3 private const String _dbName = "Northwind";
4 private Boolean _requireAbort = false;
5 private Int32 _transactionDepth = 0;
6 private Object _transactionLock = new Object();
7
8 public Database DBContext { get; protected set; }
9
10 public Guid Id { get; private set; }
11
12 public PetaPocoUnitOfWork()
13 {
14 Id = Guid.NewGuid();
15 DBContext = new Database(_dbName);
16 }
17
18 public void Begin()
19 {
20 lock (_transactionLock)
21 {
22 if (_transactionDepth == 0)
23 {
24 DBContext.BeginTransaction();
25 }
26 _transactionDepth++;
27 }
28
29 }
30
31 public void Commit()
32 {
33 lock (_transactionLock)
34 {
35 _transactionDepth--;
36 if (_transactionDepth == 0)
37 {
38 try
39 {
40 DBContext.CompleteTransaction();
41 }
42 catch
43 {
44 _transactionDepth++;
45 _requireAbort = true;
46 throw;
47 }
48 }
49 }
50 }
51
52 public void Rollback()
53 {
54 lock (_transactionLock)
55 {
56 _transactionDepth--;
57 if (_transactionDepth == 0)
58 {
59 DBContext.AbortTransaction();
60 _requireAbort = false;
61 }
62 }
63 }
64
65 public void Dispose()
66 {
67 if (_requireAbort)
68 {
69 DBContext.AbortTransaction();
70 }
71 DBContext.Dispose();
72 }
73 }

在应用层对Unity注入的IUnitOfWork调用Begin()即开启事务,对于嵌套事务变量_transactionDepth++记录事务深度,Commit()与Rollback()时_transactionDepth--,保证业务中事务只开启与提交一次。如有应用层ITradeService及实现:

1 public abstract class ApplicationService {
2 public IUnitOfWork Context { get; private set; }
3
4 public ApplicationService(IUnitOfWork context) {
5 Context = context;
6 }
7 }
8
9 public interface ITradeService {
10 void SubmitOrder(Order model);
11 }
12
13 public class TradeService : ApplicationService, ITradeService {
14 private readonly IOrderRepository _orderRepository;
15 private readonly IOrderDetailRepository _orderDetailRepository;
16
17 public TradeService(
18 IUnitOfWork context,
19 IOrderRepository orderRepository,
20 IOrderDetailRepository orderDetailRepository)
21 : base(context) {
22 _orderRepository = orderRepository;
23 _orderDetailRepository = orderDetailRepository;
24 }
25
26 void SubmitOrder(OrderDTO model){
27 //do something, like null reference check etc..
28
29 Order order = //... some logic
30 OrderDetail orderDetail = //as above
31
32 try {
33 Context.Begin();
34
35 _orderRepository.Update(order);
36 _orderDetailRepository.Update(orderDetail);
37 //could be more complex
38
39 Context.Commit();
40 }
41 catch {
42 Context.Rollback();
43 throw;
44 }
45 }
46 }

当顾客提交订单时,Context.Begin()开启一个事务锁,由于Unity的生命周期管理,当前线程内的数据上下文实例是同一个对象,故能够保证事务。当事务发生嵌套时,事务深度的解决方法发生作用,比如以可测试性的提高截取部分代码示例:

[TestClass]
public class AccountServiceTest {
[TestMethod]
public void TradeServiceTest_SubmitOrder() {
IUnitOfWork context = ... //some initialize logic
OrderDTO model = ... //as above
TradeService service = //as above context.Begin();
service.SubmitOrder();
context.Rollback();
///... etc
}
}

使用PetaPoco结合注入实现业务级事务的更多相关文章
- sql服务器第5级事务日志管理的阶梯:完全恢复模式下的日志管理
sql服务器第5级事务日志管理的阶梯:完全恢复模式下的日志管理 原文链接http://www.sqlservercentral.com/articles/Stairway+Series/73785/ ...
- spring注解实现业务层事务管理,当业务层自调用时,事务失效问题解决
前些日子一朋友在需要在目标对象中进行自我调用,且需要实施相应的事务定义,且网上的一种通过BeanPostProcessor的解决方案是存在问题的.因此专门写此篇帖子分析why. 1.预备知识 aop概 ...
- JDBC基础:JDBC快速入门,JDBC工具类,SQL注入攻击,JDBC管理事务
JDBC基础 重难点梳理 一.JDBC快速入门 1.jdbc的概念 JDBC(Java DataBase Connectivity:java数据库连接)是一种用于执行SQL语句的Java API,可以 ...
- spring项目中service方法开启线程处理业务的事务问题
1.前段时间在维护项目的时候碰到一个问题,具体业务就是更新已有角色的资源,数据库已更新,但是权限控制不起效果,还是保留原来的权限. 2.排查发现原有的代码在一个service方法里有进行资源权限表的更 ...
- servlet注入service业务bean
项目中用到spring容器来管理业务bean,在servlet中就收到前台传递来的请求参数后,调用业务bean,老是出错 部门代码如下 <span style="font-size:1 ...
- c# 业务层事务
步骤: 1.先添加System.Transactions.dll的引用 2.使用System.Transactions命名空间下的类 实例: using (TransactionScope scope ...
- Java之事务的基本应用
基本介绍 事务是数据一致性最基本的保证,也就是说一个事务中的操作要么都成功,要么都失败,不允许部分成功.我们常说的事务就是jdbc事务,当然Java中还有其他事务,并且在使用jdbc事务有很多注意点, ...
- 如何优化MySQL千万级大表
很好的一篇博客,转载 如何优化MySQL千万级大表 原文链接::https://blog.csdn.net/yangjianrong1985/article/details/102675334 千万级 ...
- Web基础之Spring AOP与事务
Spring之AOP AOP 全程Aspect Oriented Programming,直译就是面向切面编程.和POP.OOP相似,它也是一种编程思想.OOP强调的是封装.继承.多态,也就是功能的模 ...
随机推荐
- crm操作安全角色
using System; using Microsoft.Xrm.Sdk; using Microsoft.Crm.Sdk.Messages; using System.Co ...
- C++学习笔记32 断言函数
首先,让我们来看看百度百科上"断言函数"定义说明: 1定义fr=aladdin#" class="nslog:1019" title="编辑 ...
- 百度地图API详解之事件机制,function“闭包”解决for循环和监听器冲突的问题:
原文:百度地图API详解之事件机制,function"闭包"解决for循环和监听器冲突的问题: 百度地图API详解之事件机制 2011年07月26日 星期二 下午 04:06 和D ...
- 【百度地图API】如何利用PhoneGap制作地图APP
原文:[百度地图API]如何利用PhoneGap制作地图APP 摘要:百度地图API是一套由javascript编写的地图程序接口,按说它应该运行在浏览器上.现在,只要利用PhoneGap,我们就能开 ...
- NotePad++ for PHP
原文:NotePad++ for PHP 一.安装设置 1.首先根据你的系统下载相应的安装文件.http://notepad-plus-plus.org/ Notepad++插件:http://sou ...
- mvc之验证IEnumerable<T> 类型,多选框验证
原文:mvc之验证IEnumerable<T> 类型,多选框验证 假设我们有这么一种需求,我们要同时添加年级和年级下面的多个班级,我们一般会像下面这种做法. Action中我们这样接收: ...
- MyEclipse 设置全部jsp的编码为UFT-8 的方法
- awk的实施例
1.使用split功能 name.url内容: 上海 http://trip.elong.com/shanghai/jingdian elong destination 云南 http ...
- ASP.NET 5 Hello World
ASP.NET 5系列教程 (二):Hello World 本篇文章内容比较基础,主要是向大家展示如何创建一个 ASP.NET 5 工程,主要包含内容如下: 创建ASP.NET 5 工程 添加 T ...
- JSON数据转换方法 parse()和stringify()
将对象转换成JSON格式的文本数据 var str = JSON.stringify(data); 将对象转换成JSON对象的方法 var data = JSON.parse(str);