(转)设计模式之事务处理
出处:http://dennis-zane.iteye.com/blog/57553
事务处理是企业应用需要解决的最主要的问题之一。J2EE通过JTA提供了完整的事务管理能力,包括多个事务性资源的管理能力。但是大部分应用都是运行在单一的事务性资源之上(一个数据库),他们并不需要全局性的事务服务。本地事务服务已然足够(比如JDBC事务管理)。
本文并不讨论应该采用何种事务处理方式,主要目的是讨论如何更为优雅地设计事务服务。仅以JDBC事务处理为例。涉及到的DAO,Factory,Proxy,Decorator等模式概念,请阅读相关资料。
也许你听说过,事务处理应该做在service层,也许你也正这样做,但是否知道为什么这样做?为什么不放在DAO层做事务处理。显而易见的原因是业务层接口的每一个方法有时候都是一个业务用例(User Case),它需要调用不同的DAO对象来完成一个业务方法。比如简单地以网上书店购书最后的确定定单为例,业务方法首先是调用BookDAO对象(一般是通过DAO工厂产生),BookDAO判断是否还有库存余量,取得该书的价格信息等,然后调用CustomerDAO从帐户扣除相应的费用以及记录信息,然后是其他服务(通知管理员等)。简化业务流程大概如此:
首先是业务接口,针对接口,而不是针对类编程:
public interface BookStoreManager{ public boolean buyBook(String bookId,int quantity)throws SystemException; .其他业务方法}public class BookStoreManagerImpl implements BookStoreManager{ public boolean buyBook(String bookId)throws SystemException{ Connection conn=ConnectionManager.getConnection();//获取数据库连接 boolean b=false; try{ conn.setAutoCommit(false); //取消自动提交 BookDAO bookDAO=DAOFactory.getBookDAO(); CustomerDAO customerDAO=DAOFactory.getCustomerDAO(); //尝试从库存中取书 if(BookDAO.reduceInventory(conn,bookId,quantity)){ BigDecimal price=BookDAO.getPrice(bookId); //取价格 //从客户帐户中扣除price*quantity的费用 b= CustomerDAO.reduceAccount(conn,price.multiply(new BigDecimal(quantity)); . 其他业务方法,如通知管理员,生成定单等. conn.commit(); //提交事务 conn.setAutoCommit(true); } }catch(SQLException e){ conn.rollback(); //出现异常,回滚事务 con.setAutoCommit(true); e.printStackTrace(); throws new SystemException(e); } return b; } }public final class ManagerFactory { public static BookStoreManager getBookStoreManager() { return new BookStoreManagerImpl(); } }import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.sql.Connection;import com.strutslet.demo.service.SystemException;public final class TransactionWrapper { /** *//** * 装饰原始的业务代表对象,返回一个与业务代表对象有相同接口的代理对象 */ public static Object decorate(Object delegate) { return Proxy.newProxyInstance(delegate.getClass().getClassLoader(), delegate.getClass().getInterfaces(), new XAWrapperHandler( delegate)); } //动态代理技术 static final class XAWrapperHandler implements InvocationHandler { private final Object delegate; XAWrapperHandler(Object delegate) { this.delegate = delegate; } //简单起见,包装业务代表对象所有的业务方法 public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object result = null; Connection con = ConnectionManager.getConnection(); try { //开始一个事务 con.setAutoCommit(false); //调用原始业务对象的业务方法 result = method.invoke(delegate, args); con.commit(); //提交事务 con.setAutoCommit(true); } catch (Throwable t) { //回滚 con.rollback(); con.setAutoCommit(true); throw new SystemException(t); } return result; } }}public class BookStoreManagerImpl implements BookStoreManager { public boolean buyBook(String bookId)throws SystemException{ Connection conn=ConnectionManager.getConnection();// 获取数据库连接 boolean b=false; try{ BookDAO bookDAO=DAOFactory.getBookDAO(); CustomerDAO customerDAO=DAOFactory.getCustomerDAO(); // 尝试从库存中取书 if(BookDAO.reduceInventory(conn,bookId,quantity)){ BigDecimal price=BookDAO.getPrice(bookId); // 取价格 // 从客户帐户中扣除price*quantity的费用 b= CustomerDAO.reduceAccount(conn,price.multiply(new BigDecimal(quantity)); . 其他业务方法,如通知管理员,生成定单等. } }catch(SQLException e){ throws new SystemException(e); } return b; }}public final class ManagerFactory { //返回一个被包装的对象,有事务控制能力 public static BookStoreManager getBookStoreManagerTrans() { return (BookStoreManager) TransactionWrapper .decorate(new BookStoreManagerImpl()); } //原始版本 public static BookStoreManager getBookStoreManager() { return new BookStoreManagerImpl(); } }