Spring声明性事务常见问题分析

开发 后端
声明性事务是spring一个很重要的功能,可以避免开发陷入繁琐的事务控制逻辑中。但是可能是用着太方便了很多人对spring事务原理并不清楚,有必要做一番分析。

声明性事务是spring一个很重要的功能,可以避免开发陷入繁琐的事务控制逻辑中。但是可能是用着太方便了很多人对spring事务原理并不清楚,有必要做一番分析。

下边以拦截器配置方式进行说明,tx标签配置方式将在接下来另一篇文章做分析。

一、首先看配置文件:

  1. <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"> 
  2.     <property name="sessionFactory" ref="sessionFactory" /> 
  3. </bean> 
  4. <bean id="matchAllTxInterceptor" class="org.springframework.transaction.interceptor.TransactionInterceptor"> 
  5.     <property name="transactionManager"> 
  6.         <ref bean="transactionManager" /> 
  7.     </property> 
  8.     <property name="transactionAttributes"> 
  9.         <props> 
  10.             <prop key="get*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> 
  11.             <prop key="find*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> 
  12.             <prop key="search*">PROPAGATION_REQUIRED,readOnly,-Exception </prop> 
  13.             <prop key="save*">PROPAGATION_REQUIRED,-Exception </prop> 
  14.             <prop key="modify*">PROPAGATION_REQUIRED,-Exception </prop> 
  15.             <prop key="send*">PROPAGATION_REQUIRED,-Exception </prop> 
  16.             <prop key="revoke*">PROPAGATION_REQUIRED,-Exception </prop> 
  17.             <prop key="del*">PROPAGATION_REQUIRED,-Exception </prop> 
  18.             <prop key="logging*">PROPAGATION_NOT_SUPPORTED,readOnly,-Exception </prop> 
  19.             <prop key="*">PROPAGATION_SUPPORTS,-Exception </prop> 
  20.         </props> 
  21.     </property> 
  22. </bean> 
  23. <bean id="autoProxyCreator" 
  24.     class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"> 
  25.     <property name="interceptorNames"> 
  26.         <list><idref local="matchAllTxInterceptor" /></list> 
  27.     </property> 
  28.     <property name="proxyTargetClass"><value>true</value></property> 
  29.     <property name="beanNames"> 
  30.         <list><value>*Service</value></list> 
  31.     </property> 
  32. </bean> 

配置***步引入AOP代理autoProxyCreator,使用的是spring默认的jdk动态代理BeanNameAutoProxyCreator。

有两个属性要介绍一下:

1、拦截范围beanNames

例子中拦截范围是*Service,表示IOC容器中以Service结尾的bean,一般配置在spring.xml,serviceContext.xml之类的spring配置文件。

要注意这里不是值src下边的类。

bean配置信息:

  1. <bean id="menuService" class="cn.ceopen.bss..service.impl.MenuServiceImpl"/> 

有图有真相,下边是BeanNameAutoProxyCreator 调试信息。

2、截器interceptorNames

interceptorNames定义事务属性和事务管理器。

配置第二步就是定义事务属性:事务传播范围、事务隔离级别。

事务属性没什么好说的,使用spring进行事务管理的都了解,不在这里详细说了网上有大量资料。

置第三步,指定事务管理器。

这里用的是HibernateTransactionManager,spring提供对常见orm的事务支持。从spring源码可以看出HibernateTransactionManager.doGetTransaction()同时支持hibernate和jdbc。支持hibernate和jdbc混合事务,不使用jta方式的话有个前提条件:使用同一个数据源,这里所说的同一个数据源,不仅仅指物理上是同一个,在spring配置文件中也要是同一个。我在开发中遇到过这个问题,最早定义了一个数据baseDataSource,hibernate和jdbc都使用此数据源,后来项目要求使用动态数据源就又配了一个数据源dynamicDataSource仅在hibernate下做了改动,未改动jdbc对应配置,出现了事务控制问题。

出错了事务配置:

  1. <bean id="sessionFactory" 
  2. class="com.sitechasia.webx.dao.hibernate3.NamedMoudleHbmLocalSessionFactoryBean"> 
  3.     <property name="dataSource" ref="dynamicDataSource" /> 
  4.     <!--与主题无关,省略部分内容--> 
  5. </bean> 
  6. <bean id="dynamicDataSource" class="cn.ceopen.bss.pub.base.dao.RoutingDataSource">       
  7.    <property name="targetDataSources">       
  8.       <map key-type="java.lang.String">       
  9.          <entry key="baseDataSource" value-ref="baseDataSource"/>   
  10.       </map>       
  11.    </property>       
  12.    <property name="defaultTargetDataSource" ref="baseDataSource"/>       
  13. </bean> 
  14.   
  15. <bean id="baseDataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close"> 
  16.     <!--与主题无关,省略部分内容--> 
  17. </bean> 
  18. <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">  
  19.     <!--dataSource应该与sessionFactor一致--> 
  20.     <property name="dataSource"><ref bean="baseDataSource"/></property>  
  21. </bean>  
  22. <bean id="abstractJdbcDao" abstract="true"> 
  23.     <property name="jdbc" ref="jdbcTemplate" /> 
  24. </bean> 

dao配置文件:

  1. <bean id="actDao" class="cn.ceopen.bss.impl.ActDaoImpl" parent="abstractJdbcDao"/> 

dao中同时支持hibernate操作和jdbc操作。

二、事务属性传播

先看这样一个列子:

1、基于jdk动态代理的AOP事务控制,只能针对接口。

在上边的配置文件中设置的事务属性对a3()都不起作用,a3()不能单独设计事务属性,只能继承接口方法的事务属性。

2、类自身事务嵌套

***种情况:

  1. AbcIService abcService; 
  2. BcdIService bcdService; 
  3. abcService.a1(); 
  4. abcService.a2(); 
  5. bcdService.b1(); 

这三个方法对应的事务属性都起作用。

第二种情况:

方法定义

  1. public void a1() { 
  2. bcdService.b1(); 

调用:

  1. abcService.a1(); 

结果:

  1. abcService.a1(); 
  2. bcdService.b1(); 

这两个方法对应的事务属性都起作用。

第三种情况:

方法定义

  1. public void a1() { 
  2. this.a2(); 

调用:

  1. abcService.a1(); 

结果:

  1. abcService.a1(); 
  2. abcService.a2(); 

a2()对应的事务属性配置不起作用。

解决办法:

1)把a2()拆到另一个类中去;

缺点:麻烦,会把相关业务逻辑拆分了

2)调用是不用this.a2(),用abcService.a2();

  1. public void a1() { 
  2. abcService.a2(); 

缺点:要在类中注入自身引用。

原因分析:

为什么会出现这种情况呢?

我们在调用abcService.a1();时abcService是从IOC容器获取的,并AbcServiceImpl而是它的动态代理AbcServiceProxy。

示意图如下,spring不一定是这么实现的但原理一样。

AbcServiceProxy.a()方法进行了AOP增强,根据配置文件中事务属性增加了事务控制。

  1. public void a1() { 
  2. this.a2(); 

this.a2()这里this指的是AbcIServiceImpl并没用进行AOP增强,所以没用应用事务属性,只能继承a1()的事务属性。

  1. public void a1() { 
  2. abcService.a2(); 

abcService则实际是AbcServiceProxy.a2()所以可以应用事务属性。

所以在类内部进行方法嵌套调用,如果被嵌套的方法a2()需要区别于嵌套方法a1()的事务属性,需要:1)在接口公开;2)通过代理调用。

原文链接:http://www.blogjava.net/zyskm/archive/2011/11/11/363535.html

【编辑推荐】

  1. Spring MVC拦截器实现分析
  2. Spring3.0中的事务的配置方法
  3. 详解Spring中bean的作用域
  4. 详解Spring自定义属性编辑器
  5. 透视Spring定时器相关功能
责任编辑:林师授 来源: zyskm的博客
相关推荐

2009-06-22 09:01:57

Spring声明式事务

2011-05-06 15:23:46

加粉打印机

2011-05-06 15:15:21

耗材

2010-01-21 09:53:44

Cisco交换机常见问

2023-06-27 07:21:51

前端开发坑点

2011-04-08 13:58:52

JavaJSP

2011-05-30 15:12:46

电缆双绞线布线

2010-09-24 19:12:11

SQL隐性事务

2011-04-01 13:55:24

Java

2011-05-06 15:39:55

硒鼓

2013-11-14 15:47:29

SDN问题答疑

2010-07-21 09:10:02

Perl常见问题

2009-09-09 09:26:00

2010-04-07 10:42:41

Unix操作系统

2017-04-13 12:59:43

数据分析

2019-08-16 01:58:01

MySQL索引事务

2021-09-06 13:42:14

Spring声明式事务

2010-03-25 09:08:43

CentOS配置

2010-05-13 13:27:23

2009-11-02 17:25:04

ADSL常见问题
点赞
收藏

51CTO技术栈公众号