JAVA设计模式之事务处理

事务处理是企业应用需要解决的最主要的问题之一。J2EE通过JTA提供了完整的事务管理能力,包括多个事务性资源的管理能力。但是大部分应用都是运行在 单一的事务性资源之上(一个数据库),他们并不需要全局性的事务服务。本地事务服务已然足够(比如JDBC事务管理)。
    本文并不讨论应该 采用何种事务处理方式,主要目的是讨论如何更为优雅地设计事务服务。仅以JDBC事务处理为例。涉及到的 DAO,Factory,Proxy,Decorator等模式概念,请阅读相关资料。
    也许你听说过,事务处理应该做在service 层,也许你也正这样做,但是否知道为什么这样做?为什么不放在DAO层做事务处理。显而易见的原因是业务层接口的每一个方法有时候都是一个业务用例 (User Case),它需要调用不同的DAO对象来完成一个业务方法。比如简单地以网上书店购书最后的确定定单为例,业务方法首先是调用 BookDAO对象(一般是通过DAO工厂产生),BookDAO判断是否还有库存余量,取得该书的价格信息等,然后调用CustomerDAO从帐户扣 除相应的费用以及记录信息,然后是其他服务(通知管理员等)。简化业务流程大概如此:
    注意,我们的例子忽略了连接的处理,只要保证同一个 线程内取的是相同的连接即可(可用ThreadLocal实现):

    首先是业务接口,针对接口,而不是针对类编程:

  1. public  interface  BookStoreManager{
  2.           public  boolean  buyBook(String  bookId,int  quantity)throws  SystemException ;
  3.           ....其他业务方法
  4.     }


    接下来就是业务接口的实现类??业务对象:

  1. public  class  BookStoreManagerImpl implements  BookStoreManager{
  2.          public  boolean  buyBook(String  bookId)throws  SystemException {
  3.               Connection  conn=ConnectionManager.getConnection();//获取数据库连接
  4.               boolean  b=false ;
  5.              
  6.               try {
  7.                   conn.setAutoCommit(false );  //取消自动提交
  8.                   BookDAO bookDAO=DAOFactory.getBookDAO();
  9.                   CustomerDAO customerDAO=DAOFactory.getCustomerDAO();
  10.                     //尝试从库存中取书
  11.                   if (BookDAO.reduceInventory(conn,bookId,quantity)){
  12.                        BigDecimal  price=BookDAO.getPrice(bookId);  //取价格
  13.                        //从客户帐户中扣除 price*quantity的费用
  14.                        b=
  15.                        CustomerDAO.reduceAccount(conn,price.multiply(new  BigDecimal (quantity));
  16.                        ....
  17.                        其他业务方法,如通知管理员,生成定单等.
  18.                         ...
  19.                        conn.commit();   //提交事务
  20.                        conn.setAutoCommit(true );
  21.                   }
  22.                }catch (SQLException  e){
  23.                   conn.rollback();   // 出现异常,回滚事务
  24.                   con.setAutoCommit(true );
  25.                   e.printStackTrace();
  26.                   throws  new  SystemException (e);  
  27.                }
  28.                return  b;
  29.          }
  30.     }


    然后是业务代表工厂:

  1.   public  final  class  ManagerFactory {
  2.       public  static  BookStoreManager getBookStoreManager() {
  3.          return  new  BookStoreManagerImpl();
  4.       }
  5.    }


    这样的设计非常适合于DAO中的简单活动,我们项目中的一个小系统也是采用这样的设计方案,但是它不适合于更大规模的应用。首先, 你有没有闻到代码重复的 bad smell?每次都要设置AutoCommit为false,然后提交,出现异常回滚,包装异常抛到上层,写多了不烦才 怪,那能不能消除呢?其次,业务代表对象现在知道它内部事务管理的所有的细节,这与我们设计业务代表对象的初衷不符。对于业务代表对象来说,了解一个与事 务有关的业务约束是相当恰当的,但是让它负责来实现它们就不太恰当了。再次,你是否想过嵌套业务对象的场景?业务代表对象之间的互相调用,层层嵌套,此时 你又如何处理呢?你要知道按我们现在的方式,每个业务方法都处于各自独立的事务上下文当中(Transaction Context),互相调用形成了嵌 套事务,此时你又该如何处理?也许办法就是重新写一遍,把不同的业务方法集中成一个巨无霸包装在一个事务上下文中。

    我们有更为优 雅的设计来解决这类问题,如果我们把Transaction Context的控制交给一个被业务代表对象、DAO和其他Component所共知的外部 对象。当业务代表对象的某个方法需要事务管理时,它提示此外部对象它希望开始一个事务,外部对象获取一个连接并且开始数据库事务。也就是将事务控制从 service层抽离,当web层调用service层的某个业务代表对象时,返回的是一个经过Transaction Context外部对象包装(或 者说代理)的业务对象。此代理对象将请求发送给原始业务代表对象,但是对其中的业务方法进行事务控制。那么,我们如何实现此效果呢?答案是JDK1.3引 进的动态代理技术。动态代理技术只能代理接口,这也是为什么我们需要业务接口BookStoreManager的原因。
    首先,我们引入这 个Transaction Context外部对象,它的代码其实很简单,如果不了解动态代理技术的请先阅读其他资料。

  1. import  java.lang.reflect.InvocationHandler ;
  2. import  java.lang.reflect.Method ;
  3. import  java.lang.reflect.Proxy ;
  4. import  java.sql.Connection ;
  5. import  com.strutslet.demo.service.SystemException ;
  6. public  final  class  TransactionWrapper {
  7.     /**
  8.      * 装饰原始的业务代表对象,返回一个与业务代表对象有相同接口的代理 对象
  9.      */
  10.     public  static  Object  decorate(Object  delegate) {
  11.         return  Proxy .newProxyInstance(delegate.getClass().getClassLoader(),
  12.                 delegate.getClass().getInterfaces(), new  XAWrapperHandler(
  13.                         delegate));
  14.     }
  15.    
  16.     //动态代理技术
  17.     static  final  class  XAWrapperHandler implements  InvocationHandler  {
  18.         private  final  Object  delegate;
  19.         XAWrapperHandler(Object  delegate) {
  20.            this .delegate = delegate;
  21.         }
  22.        
  23.         //简单起见,包装业务代表对象所有的业务方法
  24.         public  Object  invoke(Object  proxy, Method  method, Object [] args)
  25.                 throws  Throwable  {
  26.             Object  result = null ;
  27.             Connection  con = ConnectionManager.getConnection();
  28.             try  {
  29.                 //开始一个事务
  30.                 con.setAutoCommit(false );
  31.                 //调用原始业务对象的业务方法
  32.                 result = method.invoke(delegate, args);
  33.                 con.commit();   //提交事务
  34.                 con.setAutoCommit(true );
  35.             } catch  (Throwable  t) {
  36.                 //回滚
  37.                 con.rollback();
  38.                 con.setAutoCommit(true );
  39.                 throw  new  SystemException (t);
  40.             }
  41.             return  result;
  42.         }
  43.     }
  44. }



    正如我们所见,此对象只不过把业务对象需要事务控制的业务方法中的事务 控制部分抽取出来而已。请注意,业务代表对象内部调用自身的方法将不会开始新的事务,因为这些调用不会传给代理对象。如此,我们去除了代表重复的味道。此 时,我们的业务代表对象修改成:

  1. public  class  BookStoreManagerImpl implements  BookStoreManager {
  2.     public  boolean  buyBook(String  bookId)throws  SystemException {
  3.           Connection  conn=ConnectionManager.getConnection();// 获取数据库连接
  4.           boolean  b=false ;
  5.           try {
  6.               BookDAO bookDAO=DAOFactory.getBookDAO();
  7.               CustomerDAO customerDAO=DAOFactory.getCustomerDAO();
  8.               // 尝试从库存中取书
  9.               if (BookDAO.reduceInventory(conn,bookId,quantity)){
  10.                   BigDecimal  price=BookDAO.getPrice(bookId);  // 取价格
  11.                   // 从客户帐户中扣除 price*quantity的费用
  12.                   b=
  13.                   CustomerDAO.reduceAccount(conn,price.multiply(new  BigDecimal (quantity));
  14.                   ....
  15.                   其他业务方法,如通知管理员,生成定单等.
  16.                   ...
  17.               }
  18.           }catch (SQLException  e){
  19.              throws  new  SystemException (e);
  20.           }
  21.           return  b;
  22.     }
  23.     ....
  24. }


    可以看到,此时的业务代表对象专注于实现业务逻辑,它不再关心事务控制细节,把 它们全部委托给了外部对象。业务代表工厂也修改一下,让它返回两种类型的业务代表对象:

  1. public  final  class  ManagerFactory {
  2.       //返回一个被包装的对象,有事务控制能力
  3.       public  static  BookStoreManager getBookStoreManagerTrans() {
  4.           return  (BookStoreManager) TransactionWrapper
  5.                   .decorate(new  BookStoreManagerImpl());
  6.       }
  7.       //原始版本
  8.       public  static  BookStoreManager getBookStoreManager() {
  9.          return  new  BookStoreManagerImpl();
  10.       }
  11.       ......
  12.    }


    我们在业务代表工厂上提供了两种不同的对象生成方法:一个用于创建被包装 的对象,它会为每次方法调用创建一个新的事务;另外一个用于创建未被包装的版本,它用于加入到已有的事务(比如其他业务代表对象的业务方法),解决了嵌套 业务代表对象的问题。
   我们的设计还不够优雅,比如我们默认所有的业务代表对象的方法调用都将被包装在一个 Transaction Context。可事实是很多方法也许并不需要与数据库打交道,如果我们能配置哪些方法需要事务声明,哪些不需要事务管理就更完 美了。解决办法也很简单,一个XML配置文件来配置这些,调用时判断即可。说到这里,了解spring的大概都会意识到这不正是声明式事务控制吗?正是如 此,事务控制就是AOP的一种服务,spring的声明式事务管理是通过AOP实现的。AOP的实现方式包括:动态代理技术,字节码生成技术(如 CGLIB库),java代码生成(早期EJB采用),修改类装载器以及源代码级别的代码混合织入(aspectj)等。我们这里就是利用了动态代理技 术,只能对接口代理;对类的动态代理可以使用cglib库。

猜你喜欢

转载自zhy584520.iteye.com/blog/1039147