使用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. (转)javabean操作文件正确,但是Jsp调用javabean时文件路径出错问题解决之JavaBean访问本地文件实现路径无关实现方法

        在JSP中,页面链接是使用web路径的,但如果JavaBean要访问本地文件读取配置信息的话,是需要文件的本地路径的.如果你在写 Bean的时候直接将本地路径写进去,那网站的路径就不能变化,丧 ...

  2. 浅谈 js 字符串 trim 方法之正则篇

    原文:浅谈 js 字符串 trim 方法之正则篇 关于 trim 其实没啥好说的,无非就是去除首位空格,对于现代浏览器来说只是简单的正则 /^\s+|\s+$/ 就可以搞定了.而且支持中文空格   等 ...

  3. ZOJ 2724 Windows 消息队列 (优先队列)

    链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2724 Message queue is the basic fund ...

  4. The Swift Programming Language-官方教程精译Swift(8)闭包 -- Closures

    闭包是功能性自包含模块,可以在代码中被传递和使用. Swift 中的闭包与 C 和 Objective-C中的 blocks 以及其他一些编程语言中的 lambdas 比较相似. 闭包可以捕获和存储其 ...

  5. EditPlus仿Sublime配色方案(三种任你选择)

    [Options] Placement=2C00000002000000030000000083FFFF0083FFFFFFFFFFFFFFFFFFFFEA000000EE000000D6050000 ...

  6. MySQL在Windows和Linux减少数据库

    Linux减少数据库代码: 1,创建一个空数据库cddl mysql> create database cddl; Query OK, 1 row affected (0.00 sec) 2,还 ...

  7. 重新想象 Windows 8.1 Store Apps (81) - 控件增强: 加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Contract 分享 WebView 中的内容, 为 WebView 截图

    原文:重新想象 Windows 8.1 Store Apps (81) - 控件增强: 加载本地 html, 智能替换 html 中的 url 引用, 通过 Share Contract 分享 Web ...

  8. Floodlight controller 线程池模型

         官方文档对于ThreadPool的描写叙述是:ThreadPool is a Floodlight module wrapper for a Java's ScheduledExecutor ...

  9. 删除Python UserWarning[已解决]

    在使用MySQLdb包后,导入测试时发现一个警告. /usr/lib/python2.6/site-packages/setuptools-0.8-py2.6.egg/pkg_resources.py ...

  10. [ACM] POJ 2506 Tiling (递归,睑板)

    Tiling Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 7487   Accepted: 3661 Descriptio ...