`

设计模式之事务处理

阅读更多
    事务处理是企业应用需要解决的最主要的问题之一。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();
      }

   }



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

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

      
   }

   
   我们在业务代表工厂上提供了两种不同的对象生成方法:一个用于创建被包装的对象,它会为每次方法调用创建一个新的事务;另外一个用于创建未被包装的版本,它用于加入到已有的事务(比如其他业务代表对象的业务方法),解决了嵌套业务代表对象的问题。
   我们的设计还不够优雅,比如我们默认所有的业务代表对象的方法调用都将被包装在一个Transaction Context。可事实是很多方法也许并不需要与数据库打交道,如果我们能配置哪些方法需要事务声明,哪些不需要事务管理就更完美了。解决办法也很简单,一个XML配置文件来配置这些,调用时判断即可。说到这里,了解spring的大概都会意识到这不正是声明式事务控制吗?正是如此,事务控制就是AOP的一种服务,spring的声明式事务管理是通过AOP实现的。AOP的实现方式包括:动态代理技术,字节码生成技术(如CGLIB库),java代码生成(早期EJB采用),修改类装载器以及源代码级别的代码混合织入(aspectj)等。我们这里就是利用了动态代理技术,只能对接口代理;对类的动态代理可以使用cglib库。
    这篇短文只是介绍下我对事务上下文模式以及声明式事务管理实现基本原理的理解,如有错误,请不吝赐教。我的email:killme2008@gmail.com
分享到:
评论
2 楼 martintree 2010-11-15  
此文我转走了,谢谢
1 楼 cloud21 2009-09-23  
Have dinner Every day,It is my work.

skimming Paper of yours  Every day,It is my mission.

相关推荐

    JAVA设计模式之事务处理

    "JAVA设计模式之事务处理"主要关注如何在业务逻辑中有效地管理和控制事务。 事务处理在企业级应用中至关重要,因为它确保数据的一致性和完整性。Java平台提供了Java Transaction API (JTA) 来处理全局事务,适用...

    JAVA设计模式之事务处理.pdf

    ### JAVA设计模式之事务处理 #### 一、引言 在企业级应用开发中,事务处理是一项关键的技术,尤其在涉及多个数据操作时尤为重要。Java技术体系中的J2EE(Java 2 Enterprise Edition)提供了强大的事务管理机制——...

    JAVA设计模式之事务处理[收集].pdf

    JAVA设计模式之事务处理[收集].pdf

    JAVA设计模式之事务处理.docx

    Java设计模式中的事务处理是构建企业级应用时不可或缺的一部分,主要关注数据的一致性和完整性。在J2EE环境中,Java Transaction API (JTA) 提供了全面的事务管理能力,适用于涉及多个事务资源的情况。然而,对于...

    Java事务设计模式_java_事务设计模式_

    以上内容主要围绕Java中的事务处理策略展开,包括事务的基本概念、不同的事务管理模型以及事务设计模式的应用。了解和掌握这些知识,对于开发高可靠性的Java应用至关重要。在实际项目中,应根据系统需求和复杂度选择...

    java事务设计模式

    Java事务设计模式是系统开发中不可或缺的一部分,尤其是在大型分布式系统中,确保数据的一致性和完整性至关重要。本设计模式主要关注如何在Java环境中有效地管理和协调事务,以满足ACID(原子性、一致性、隔离性、...

    dao设计模式DAO 设计模式之事务界定疯.pdf

    ### DAO设计模式与事务管理 #### 一、引言 在软件开发中,尤其是在基于数据库的应用程序中,数据访问层(Data Access Object, DAO)设计模式是一种常用的模式,用于简化和标准化对数据库的操作。该模式的主要目的...

    java事务 - 模板设计模式

    而模板设计模式则是一种行为设计模式,它提供了一种方法的通用实现,允许子类在不改变整体结构的情况下,定制特定的步骤。当我们将这两种概念结合时,我们可以通过Template模式来优雅地处理事务的控制。 首先,让...

    设计模式之禅 设计模式

    它在客户端和目标对象之间起到中介作用,可以用于延迟加载、安全控制或事务处理等场景。 7. **适配器模式**:将一个类的接口转换成客户希望的另一个接口。适配器使原本由于接口不兼容而不能一起工作的那些类可以...

    新版设计模式手册 - C#设计模式(第二版)

    设计模式是解决软件开发中常见问题的经验总结,是软件工程的最佳实践之一。本手册将详细阐述23种经典设计模式,并结合C#语言特性,给出具体实现和应用示例。 首先,我们来了解一下设计模式的基本概念。设计模式是...

    DAO设计模式 DAO 设计模式 JAVA设计模式

    DAO(Data Access Object)设计模式是软件开发中一种常见的用于处理数据访问的模式,它将业务逻辑与数据访问逻辑分离,使得代码结构更加清晰,更易于维护和测试。在Java开发中,DAO模式广泛应用于数据库操作,为应用...

    设计模式——刘伟

    设计模式是软件工程中的一种重要概念,它代表了在特定情境下解决常见问题的最佳实践。刘伟先生在讲解设计模式时,通常会深入浅出地介绍这些模式的原理、应用场景以及如何有效地在实际编程中应用它们。设计模式并不是...

    联机事务处理系统

    联机事务处理系统(OLTP)是数据库管理的核心组件之一,主要用于处理大量的事务性操作,如银行交易、订单处理和机票预订等。这些系统旨在提供高并发、低延迟的数据处理能力,确保数据的一致性和完整性。OLTP系统可以...

    设计模式之禅之 工厂方法模式+策略模式

    ### 设计模式之禅之 工厂方法模式+策略模式 #### 一、概述 《设计模式之禅》第35章介绍了工厂方法模式与策略模式的结合运用,旨在通过一个具体的案例——“一卡通”交易系统的设计,来展现如何在实际项目中灵活...

    设计模式,分布式设计模式,Java设计模式

    设计模式是软件工程中的一种最佳实践,它是在特定情境下为了解决常见问题而形成的可重用解决方案。设计模式不仅仅是代码结构,更是经验的总结,是解决特定问题的有效方法论。在Java这样的面向对象编程语言中,设计...

    Socket_udp多事务处理程序框架

    ### Socket_UDP多事务处理...综上所述,Socket_UDP多事务处理程序框架在设计时需充分考虑UDP协议的特点,通过合理的设计和优化策略,解决无连接状态下的数据包管理和事务处理难题,以实现高效、稳定的网络通信服务。

    【设计模式】之责任链模式1

    责任链模式是一种行为设计模式,它允许将请求沿着处理者对象的链进行发送,而接收者可以在链中的任何位置处理该请求。这种模式的核心思想是将处理请求的对象组织成一个链条,每个对象(节点)都包含对下个处理者的...

    事务处理与云计算的文章

    而云计算则是一种分布式计算模式,通过互联网提供按需计算服务,包括服务器、存储、数据库、网络、软件、分析等,极大地扩展了事务处理的能力和范围。 事务处理的核心在于ACID(原子性、一致性、隔离性和持久性)...

    命令模式实现事务补偿2

    在分布式系统设计中,事务处理是一项关键任务,它涉及到数据的一致性和完整性。"命令模式实现事务补偿2"是一个解决方案,旨在解决跨平台多原子服务协调调用中的异常处理问题。这种实现方式利用了设计模式中的命令...

Global site tag (gtag.js) - Google Analytics