首页 | 互联网 | IT动态 | IT培训 | Cisco | Windows | Linux | Java | .Net | Oracle | 软件测试 | C/C++ | 嵌入式开发 | 存储世界 | 服务器
网络设备 | IDC | 安全 | 求职招聘 | 数字网校 | 网页设计 | 平面设计 | 技术专题 | 电子书下载 | 教学视频 | 源码下载 | 搜索 | 博客 | 论坛
中国IT实验室Linux频道
中国IT教育
Google
首页 资讯动态 认证考试 新手入门 核心技术 高级技术 J2EE J2ME Java&XML 开源技术 其他技术 RSS订阅 论坛 专题
您现在的位置: 中国IT实验室 >> Java >> 高级技术 >> 设计模式 >> 正文

JAVA设计模式之事务处理


 

  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库。
    这篇短文只是介绍下我对事务上下文模式以及声明式事务管理实现基本原理的理解,如有错误,请不吝赐教,谢谢。我的email:killme2008@gmail.com

上一页  [1] [2] 

【责编:Peng】

中国IT教育

相关产品和培训
文章评论
 友情推荐链接
 认证培训
 专题推荐

 ·关于Java框架技术专题
 ·XML全攻略技术专题
 ·JAVA开源技术介绍专题
 ·Java嵌入式开发之J2ME技术专题
 ·超前体验 Oracle 11g的5个新特性…
 ·揭密使用VB.NET的五个实用技巧
 ·Oracle和SQL Server常用函数对比专题…
 ·展现C#世界 C#程序设计专题…
 ·Java入门 Tomcat的配置技巧精华专题…
 ·Oracle RMAN物理备份技术详解…
 今日更新
 社区讨论
 博客论点
 频道精选
 Java 频道导航