使用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
}
}
 
 
 
标签: .Net

使用PetaPoco结合注入实现业务级事务的更多相关文章

  1. sql服务器第5级事务日志管理的阶梯:完全恢复模式下的日志管理

    sql服务器第5级事务日志管理的阶梯:完全恢复模式下的日志管理 原文链接http://www.sqlservercentral.com/articles/Stairway+Series/73785/ ...

  2. spring注解实现业务层事务管理,当业务层自调用时,事务失效问题解决

    前些日子一朋友在需要在目标对象中进行自我调用,且需要实施相应的事务定义,且网上的一种通过BeanPostProcessor的解决方案是存在问题的.因此专门写此篇帖子分析why. 1.预备知识 aop概 ...

  3. JDBC基础:JDBC快速入门,JDBC工具类,SQL注入攻击,JDBC管理事务

    JDBC基础 重难点梳理 一.JDBC快速入门 1.jdbc的概念 JDBC(Java DataBase Connectivity:java数据库连接)是一种用于执行SQL语句的Java API,可以 ...

  4. spring项目中service方法开启线程处理业务的事务问题

    1.前段时间在维护项目的时候碰到一个问题,具体业务就是更新已有角色的资源,数据库已更新,但是权限控制不起效果,还是保留原来的权限. 2.排查发现原有的代码在一个service方法里有进行资源权限表的更 ...

  5. servlet注入service业务bean

    项目中用到spring容器来管理业务bean,在servlet中就收到前台传递来的请求参数后,调用业务bean,老是出错 部门代码如下 <span style="font-size:1 ...

  6. c# 业务层事务

    步骤: 1.先添加System.Transactions.dll的引用 2.使用System.Transactions命名空间下的类 实例: using (TransactionScope scope ...

  7. Java之事务的基本应用

    基本介绍 事务是数据一致性最基本的保证,也就是说一个事务中的操作要么都成功,要么都失败,不允许部分成功.我们常说的事务就是jdbc事务,当然Java中还有其他事务,并且在使用jdbc事务有很多注意点, ...

  8. 如何优化MySQL千万级大表

    很好的一篇博客,转载 如何优化MySQL千万级大表 原文链接::https://blog.csdn.net/yangjianrong1985/article/details/102675334 千万级 ...

  9. Web基础之Spring AOP与事务

    Spring之AOP AOP 全程Aspect Oriented Programming,直译就是面向切面编程.和POP.OOP相似,它也是一种编程思想.OOP强调的是封装.继承.多态,也就是功能的模 ...

随机推荐

  1. Android基础之——MacOSX下elipse开发环境的配置

    前不久换了台macbook,然后自己就把开发环境给配好了,本来这事就这么过去了,今天有位博友留言让我写一篇关于配置的文章,考虑到这个东西确实以后可能还会用,那就写下来,分享给大家,正好自己也再次回想一 ...

  2. 使用myeclipse创建带注解的model实体类

    1.先新建JPA项目: 如果没有就点击左下角的Show All Wizards. 点两次Next后,点击Finish即可,中间不用任何操作 (点第二次Next后会出现连接到所在数据库,先不管)     ...

  3. Java多线程的~~~Lock接口和ReentrantLock使用

    在多线程开发.除了synchronized这个keyword外,我们还通过Lock接口来实现这样的效果.由Lock接口来实现 这样的多线程加锁效果的优点是非常的灵活,我们不在须要对整个函数加锁,并且能 ...

  4. C#函数式编程-序列

    C#函数式编程之序列 过了许久的时间,终于趁闲暇的时间来继续将函数式编程这个专辑连载下去,这段时间开头是为IOS这个新方向做准备,将OC的教程写成了SWIFT版,当然我个人是支持Xamarin,但是我 ...

  5. SQL表连接

    背景 在上次的自考科目<数据库系统原理>中.已经接触到了关于数据库表连接的一些知识,近期的学习过程中又用到了关于数据库表的连接问题,趁此再跟大家一起回想一下. 导图总结 首先用一张思维导图 ...

  6. style中position的属性值具体解释

    Position的英文原意是指位置.职位.状态.也有安置的意思.在CSS布局中,Position发挥着非常关键的数据,非常多容器的定位是用Position来完毕. Position属性有四个可选值,它 ...

  7. Dungeon Master poj 2251 dfs

    Language: Default Dungeon Master Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 16855 ...

  8. MVC验证02-自定义验证规则、邮件验证

    原文:MVC验证02-自定义验证规则.邮件验证 本文体验MVC自定义验证特性,来实现对邮件的验证.对于刚写完的自定义验证特性,起初只能支持后端验证.如果要让前端jquery支持,还必须对jquery的 ...

  9. 找不到方法: Int32 System.Environment.get_CurrentManagedThreadId() .

    这个问题在本地运行没错...放到服务器上就出现这个问题.. 原因:是这个方法是.NETFRAMWORK4.5的..服务器上用的是4.0就会出现这个问题. 解决办法:在本地WEB项目右键把项目改到FRA ...

  10. hdu oj1102 Constructing Roads(最小生成树)

    Constructing Roads Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...