对spring中AOP的理解(事务管理)

来源:互联网 发布:人员优化方案 编辑:程序博客网 时间:2024/05/21 02:34
你听说过 EJB 2.0 里面的 EJB 是如何处理事务的么?如果你下载个 EJB 2.0 规范的 PDF 看一下几种 transaction attribute 设定下事务的最终结果是什么样的,更重要的是 RemoteException 和 RuntimeException 为什么叫系统异常,系统异常导致 EJB 服务器自动回滚事务,而应用程序异常都表示应用程序自己来决定是否回滚异常(通过 mySessionContext.setRollbackOnly() 来回滚事务)。EJB 的事务是在服务器内部实现是根据你的transaction attribute 的设定是否要给当前线程绑定一个 transaction (通过 ThreadLocal.set($transaction)),如果碰到 requiresNew 则把原来 ThreadLocal.get() 得到的压入堆栈,重新 TheadLocal.set 另一个。而在EJB调用过程中的事务则是通过 ThreadLocal.get() 得到当前的事务。这些服务器内部的做法我们不需要知道,只需要知道事务是通过与线程绑定在一起的,根据 transaction attribute 来决定当前绑定的是哪个事务,与调用者的事务是共用一个还是新开另一个。知道事务的线程绑定和 EJB 中异常的分类就明白 EJB 是如何管理事务的。而 Spring 的事务管理其实也是类似的,它也是使用类似的将事务绑定在当前调用的线程搜索,然后也根据你声明的什么 exception 需要回滚事务来决定的事务的提交还是回滚。
提问者评价
谢谢!

怎样通过spring进行事务管理呢?下面进行简单的介绍~

假设我们现在有一个接口叫IBusinessService,里面有一个方法叫doSaveInformation(),有一个实现这个接口的类叫BusinessService,在doSaveInformation中有一系列的数据库操作,如下:

public interface IBusinessService{

public void doSaveInformation();

}

public class BusinessService implements IBusinessService{

public void doSaveInformation(){

进行一系列数据库操作;

}

}

我们想通过spring对这个方法进行事务管理,把这个方法当做一个事务去处理,该怎么做呢?

首先,在spring上下文中定义业务类:

<bean id="businessService" class="BusinessService">

</bean>

其次,为spring上下文添加拦截器:

?<!--定义拦截的类--> <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> <property name="proxyTargetClass"> <value>true</value> </property> <property name="beanNames"> <value>*Service</value> </property> <property name="interceptorNames"> <list> <value>transactionInterceptor</value> </list> </property> </bean>

再次,定义拦截器需要管理的事务(假设sessionFactory以及数据源再spring上下文中已经有定义):<!--定义Hibernate 的 事务--> <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> <property name="sessionFactory"> <ref local="sessionFactory"/> </property> </bean>

最后,定义拦截器的拦截策略:<!--定义事务拦截器的拦截策略--> <bean id="transactionInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor"> <property name="transactionManager" ref="transactionManager"/> <property name="transactionAttributes"> <props> <prop key="doSave*">PROPAGATION_REQUIRED,-BusinessException</prop> </props> </property> </bean>

在java代码中通过getBean("businessService")来获得业务类,然后调用业务类的doSaveInformation()方法,这样spring就能对BusinessService类中的doSaveInformation()方法进行拦截,把这个方法作为一个事务进行管理了。

spring是怎样做的呢?

此处进介绍spring通过接口方式实现的代理模式

这里做一个浅显的解释,在通过getBean("businessService")获得的业务类时,实际上得到的是一个代理类,假设这个类叫做BusinessServiceProxy:

?

注意,spring已经获取了上述的配置,并在下面生成的代理类的doSaveInformation()方法中添加了事务管理的代码(上文中配置的含义是,针对类名符合“*Service”且类中的方法名符合“doSave*”的方法进行事务包裹,并且当在方法中捕获到BusinessException时,事务回滚)

public class BusinessServiceProxy implements IBusinessService{

private IBusinessService businessService;

BusinessServiceProxy(IBusinessService businessService){

this.businessService = businessService;

}

public void doSaveInformation(){

try{

启动事务的代码;businessService.doSaveInformation();

事务提交的代码;

}catch(BusinessExcepiton e){

回滚事务的代码;

}

}

}

也就是说实际执行的代码是BusinessServiceProxy.doSaveInformation(),自然就是带事务管理的了~

?

在项目中编写代码是有一个常见的错误,我们会发现,如果这样做,spring的事务管理不管用,错误实例如下:

还那上面的类和方法来说,如果做一点小小的变动,在BusinessService类中增加一个方法,doProcessAndSave(),如下:

public interface IBusinessService{

public void doProcessAndSave();

public void doSaveInformation();

}

public class BusinessService implements IBusinessService{

public void doProcessAndSave(){

进行一些业务处理操作;

this.doSaveInformation();

}

public void doSaveInformation(){

进行一系列数据库操作;

}

}

这是,我们同样通过spring的getBean("businessService")获得的业务类,然后调用该类的doProcessAndSave()方法,在这个方法中我们调用了doSaveInformation()方法,我们发现,doSaveInformation()的事务失效了!

其实原理一样,spring一样构造了一个代理类,在代理类中调用了doProcessAndSave()而不是doSaveInformation()方法,spring会发现我们并没有对doProcessAndSave()方法进行事务包裹,因此spring所生成的代理类是不带事务的。所以想把doSaveInformation()方法当做一个事务使用时,需要直接通过spring创建的代理类去调用该方法,而非套在其他的方法中。

?

欢迎大家指正错误~

MSN空间完美搬家到新浪博客!

  • Struts2标签介绍2

    2010-10-18 2:31:00

    ? / ?*?$Id:?UITagExample.java?420385?2006-07-10?00:57:05Z?mrdon?$ ?* ?*?Copyright?2006?The

  • hibernate 主键生成策略

    2011-10-5 13:36:00

    @Id ?@GeneratedValue(generator="UserInfoId") ?@GenericGenerator(name="UserInfoId&quo

  • hibernate c3p0 配置

    2010-10-26 3:33:00

    前几天写过一篇这样的日志,因为写的时候我也没太弄明白,所以后来就又接着查资料,把这几天了解到的再记录一下 Hibernate 里可以设置的属性不多: #c3p0-native proper

  • 关于HibernateDaoSupport的getHibernateTemplate().find

    2011-8-17 14:42:00

    在 hibernate 和spring的整合应用中,SessionFactory是使用spring提供的。即我们只要在Dao中继承HibernateDaoSupport就可以使用,一般使用get

  • 引用 Tomcat配置虚拟主机

    2010-10-7 9:21:00

    ? 引用 【轰隆隆】 的 Tomcat 配置虚拟主机 一、打开 tomcat 安装目录下conf/server.xml这个文件 在server.xml文档中找

  • linux集群LVS+GFS+ISCSI+TOMCAT

    2011-2-18 21:01:00

    我是从最初的HA(高可用性)开始的,别人的例子是用VMWARE,可以做试验但不能实际应用,我又没有光纤卡的Share Storage,于是就选用ISCSI,成功后又发现ISCSI+EXT3不能用于L

  • 错误:Target runtime apache tomcat v6.0 is not defined

    2011-7-24 10:51:00

    在“错误”视图中直接右击删除出错的条目就可以了。 如果不行,还可以试试如下方法: 在Eclipse工程目录下的.settings文件夹里,打开org.eclipse.wst.common.pro

  • spring定时任务

    2010-10-26 11:35:00

    什么是动态定时任务:是由客户制定生成的,服务端只知道该去执行什么任务,但任务的定时是不确定的(是由客户制定)。 这样总不能修改配置文件每定制个定时任务就增加一个trigger吧,即便允许客户修改配

  • Struts2.0 配置和一些值得注意的地方

    2011-3-12 8:28:00

    0. action的基本配置(我用的是 struts 2.2.1.1 Struts 2.2.1.1 is the best available version of Struts

  • 用Hibernate进行分页显示时每次显示的图书列表的顺序不一样

    2011-10-10 20:22:00

    下面简要的表示一下逻辑: 有Category和Product存在多对多的关联,Book实体继承了Product public class Category implements java.io

0 0
原创粉丝点击