spring的事物配置方式

本文介绍五种Spring事务管理配置方式,包括为每个bean设置代理、所有bean共享一个代理、使用拦截器、AOP配置及注解方式。每种方式详细展示了配置示例,帮助读者理解并实践。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

 

根据代理机制不同,有以下几种配置方式:(推荐使用第四种方式,做spring的事物)

 

首先定义一个接口和实现类,并在此基础上进行配置---

Java代码  

收藏代码

public interface IUserDao {  
 public void insertUser(UserTable user);  
}  
 
 

Java代码  

收藏代码

public class UserDaoImpl extends HibernateDaoSupport implements IUserDao{  
    public void insertUser(UserTable user) {  
        getHibernateTemplate().saveOrUpdate(user);  
    }  
}  
 
 

 

  • 第一种:每个bean设置一个代理,这种是根据具体需求来定,如要对具体到每个交易进行事务操作的话,这个方式是最合适的

 

<!-- 每个bean都有一个代理 -->  

<beans>  
    <!-- sessionFactory相当于spring datasource -->  
    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">    
        <property name="configLocation" value="classpath:hibernate.cfg.xml" />    
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />  
    </bean>    
    <!-- 定义事务管理器(声明式的事务) -->    
    <bean id="transactionManager"  
        class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
      
    <!-- 配置DAO -->  
    <bean id="userDao" class="org.lgh.spring.transaction2.UserDaoImpl">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
    <!-- 每个bean都有一个代理 <property name="target" ref="userDaoTarget" />  -->  
    <bean id="userDaoProxy"    
        class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">    
           <!-- 配置事务管理器 -->    
           <property name="transactionManager" ref="transactionManager" />       
        <property name="target" ref="userDao" />    
         <property name="proxyInterfaces" value="org.lgh.spring.transaction2.IUserDao" />  
        <!-- 配置事务属性 -->    
        <property name="transactionAttributes">    
            <props>    
                <prop key="*">PROPAGATION_REQUIRED</prop>  
            </props>    
        </property>    
    </bean>    
</beans>  
 

 

  • 第二种:所有的bean共享一个代理

 

 <!-- 所有的bean共享一个代理-->  
<beans>  
    <bean id="sessionFactory"    
            class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">    
        <property name="configLocation" value="classpath:hibernate.cfg.xml" />    
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />  
    </bean>    
    <!-- 定义事务管理器(声明式的事务) -->    
    <bean id="transactionManager"  
        class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
    <!-- 所有的bean共享一个代理/>  -->  
    <bean id="transactionBase"    
            class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"    
            lazy-init="true" abstract="true">    
        <!-- 配置事务管理器 -->    
        <property name="transactionManager" ref="transactionManager" />    
        <!-- 配置事务属性 -->    
        <property name="transactionAttributes">    
            <props>    
                <prop key="*">PROPAGATION_REQUIRED</prop>    
            </props>    
        </property>    
    </bean>      
    <!-- 配置DAO -->  
    <bean id="userDao" class="org.lgh.spring.transaction3.UserDaoImpl">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
     <!-- 所有的bean共享一个代理/>  -->  
    <bean id="userDaoProxy" parent="transactionBase" >    
        <property name="target" ref="userDao" />     
    </bean>  
</beans>  
 

 

  • 第三种:使用拦截器 来配置你的事务,这个主要是进行一些方法调用前后进行一些其他事件的处理,如进行权限检查等...

 

<!-- 使用拦截器 -->  
  
<beans>  
    <bean id="sessionFactory"    
            class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">    
        <property name="configLocation" value="classpath:hibernate.cfg.xml" />    
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />  
    </bean>    
    <!-- 定义事务管理器(声明式的事务) -->    
    <bean id="transactionManager"  
        class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>   
    <bean id="transactionInterceptor"    
        class="org.springframework.transaction.interceptor.TransactionInterceptor">    
        <property name="transactionManager" ref="transactionManager" />    
        <!-- 配置事务属性 -->    
        <property name="transactionAttributes">    
            <props>    
                <prop key="*">PROPAGATION_REQUIRED</prop>    
            </props>    
        </property>    
    </bean>  
    <bean name="logger" class="org.lgh.spring.transaction4.SpringAOPInterceptor">  
    </bean>  
    <bean id="logBeforeAdvice" class="com.spring.advices.LogBeforeAdvice">  
    </bean>  
    <!-- 定义BeanNameAutoProxyCreator,该bean是个bean后处理器,无需被引用,因此没有id属性     
    这个bean后处理器,根据事务拦截器为目标bean自动创建事务代理    -->  
    <bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator">    
        <!-- 加上此句就不会抛出 $Proxy cannot be cast的异常啦-->  
        <property name="proxyTargetClass">  
              <value>true</value>  
        </property>    
        <property name="beanNames">    
            <list>    
            <!-- *Dao对应下面的userDao要对它进行拦截-->  
                <value>userDao</value>  
            </list>    
        </property>    
        <property name="interceptorNames">    
            <list>    
                <value>transactionInterceptor</value>    
                <value>logBeforeAdvice</value>  
            </list>    
        </property>    
    </bean>    
    <!-- 配置DAO -->  
    <bean id="userDao" class="org.lgh.spring.transaction4.UserDaoImpl">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
    <!-- 服务层 service-->  
     <bean id="userService" class="org.lgh.spring.transaction4.UserServiceImpl">  
        <property name="userDao" ref="userDao" />  
    </bean>  
</beans>  
 

 

  • 第四种:使用aop:config配置方式

 

<beans xmlns="http://www.springframework.org/schema/beans"  
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
    xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:aop="http://www.springframework.org/schema/aop"  
    xmlns:tx="http://www.springframework.org/schema/tx"  
    xsi:schemaLocation="http://www.springframework.org/schema/beans   
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
           http://www.springframework.org/schema/context  
           http://www.springframework.org/schema/context/spring-context-2.5.xsd  
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">  
 <context:annotation-config />  
    <context:component-scan base-package="org.lgh.spring.transaction5" />  
    <bean id="sessionFactory"    
            class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">    
        <property name="configLocation" value="classpath:hibernate.cfg.xml" />    
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />  
    </bean>    
    <!-- 定义事务管理器(声明式的事务) -->    
    <bean id="transactionManager"  
        class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
    <tx:advice id="txAdvice" transaction-manager="transactionManager">  
        <tx:attributes> 
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="delete*" propagation="REQUIRED" />
 <tx:method name="update*" propagation="REQUIRED" />
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="*" propagation="REQUIRED" />  
</tx:attributes>  
    </tx:advice>  
    <aop:config   proxy-target-class="true" >  
        <aop:pointcut id="interceptorPointCuts"  
            expression="execution(* org.lgh.spring.transaction5.*.*(..))" />  
        <aop:advisor advice-ref="txAdvice"  
            pointcut-ref="interceptorPointCuts" />          
    </aop:config>       
    <!-- 配置DAO -->  
    <bean id="userDao" class="org.lgh.spring.transaction5.UserDaoImpl">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
</beans>  
 

 

 

execution(* org.lgh.spring.transaction5.*.*(..))中:

第一个 * —— 通配 任意返回值类型|

|第二个 * —— 通配 包org.lgh.spring.transaction5下的任意class|

|第三个 * —— 通配 包org.lgh.spring.transaction5下的任意class的任意方法|

|第四个 .. —— 通配 方法可以有0个或多个参数|

 

  •  第五种:注解方式:

 

<beans xmlns="http://www.springframework.org/schema/beans"  
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
    xmlns:context="http://www.springframework.org/schema/context"  
    xmlns:aop="http://www.springframework.org/schema/aop"  
    xmlns:tx="http://www.springframework.org/schema/tx"  
    xsi:schemaLocation="http://www.springframework.org/schema/beans   
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
           http://www.springframework.org/schema/context  
           http://www.springframework.org/schema/context/spring-context-2.5.xsd  
           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
           http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">  
 <context:annotation-config />  
    <context:component-scan base-package="org.lgh.spring.transaction6" />  
    <tx:annotation-driven proxy-target-class ="true" transaction-manager="transactionManager"/>   
    <bean id="sessionFactory"    
            class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">    
        <property name="configLocation" value="classpath:hibernate.cfg.xml" />    
        <property name="configurationClass" value="org.hibernate.cfg.AnnotationConfiguration" />  
    </bean>    
    <!-- 定义事务管理器(声明式的事务) -->    
    <bean id="transactionManager"  
        class="org.springframework.orm.hibernate3.HibernateTransactionManager">  
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
    <!-- 配置DAO -->  
    <bean id="userDao" class="org.lgh.spring.transaction6.UserDaoImpl">  
          
        <property name="sessionFactory" ref="sessionFactory" />  
    </bean>  
      
</beans>  
<think>好的,用户现在问的是Spring事务失效的几种方式。之前他们问过类似的问题,但可能没有得到足够详细的回答,或者他们现在想深入理解。我需要先回忆一下常见的Spring事务失效的场景,然后组织成清晰的点。 首先,用户可能是开发中遇到了事务没有按预期工作的情况,所以想知道哪些情况会导致事务失效。常见的可能有方法内部调用、异常处理不当、事务传播设置错误等等。 然后,我需要确保每个失效场景都解释清楚,比如为什么内部调用会导致事务失效,因为Spring的事务是基于代理的,自调用时代理不会介入。还有异常类型的问题,比如默认只回滚RuntimeException,而checked异常不会触发回滚,除非特别声明。 另外,用户可能对事务传播机制不太熟悉,比如REQUIRED和REQUIRES_NEW的区别,或者在不同方法中使用不同传播行为导致的问题。需要解释清楚每个传播行为的含义和可能的影响。 还要考虑数据库引擎是否支持事务,比如MyISAM不支持,而InnoDB支持。这点容易被忽略,但也是常见的问题来源。 还有是否在非public方法上使用事务,因为SpringAOP默认只对public方法生效。如果用户在一个protected或private方法上加了@Transactional,事务也不会生效。 另外,多线程环境下的事务问题,比如新开的线程中的操作不在同一个事务里,这也是需要注意的地方。 可能用户还不太清楚如何正确配置事务管理器,或者在不同的数据源下没有正确配置,导致事务管理器无法正常工作。 最后,需要给出一些排查建议,比如检查代理是否生效,异常是否被捕获,日志级别调整等,帮助用户在实际开发中快速定位问题。 总结起来,应该分点列出每种失效场景,给出原因和解决方案,结构清晰,语言简洁。避免使用太专业的术语,但必要的术语需要解释。这样用户能一步步排查自己的问题,理解每个可能导致事务失效的原因。</think>Spring 事务失效的常见场景常与事务的配置、代理机制、异常处理或数据库引擎支持有关。以下是几种典型情况: --- ### 1. **方法内部调用(自调用问题)** - **原因**:Spring 事务基于 AOP 代理实现,当在一个类的内部方法(如方法 A)中直接调用另一个事务方法(如方法 B)时,事务 B 不会生效(绕过代理,直接调用原始对象的方法)。 - **示例**: ```java @Service public class UserService { public void methodA() { // 直接调用内部方法,事务失效 methodB(); } @Transactional public void methodB() { // 数据库操作 } } ``` - **解决**: - 将事务方法拆分到不同类中。 - 过 `AopContext.currentProxy()` 获取代理对象再调用。 --- ### 2. **异常未正确抛出** - **原因**:Spring 默认只在抛出 `RuntimeException` 或 `Error` 时回滚事务。如果捕获了异常未抛出,或抛出了 `checked` 异常(如 `Exception`),事务不会回滚。 - **示例**: ```java @Transactional public void update() { try { // 数据库操作 } catch (Exception e) { // 捕获异常未抛出,事务不回滚 } } ``` - **解决**: - 抛出 `RuntimeException` 或配置 `@Transactional(rollbackFor = Exception.class)`。 - 避免在事务方法内捕获异常后不抛出。 --- ### 3. **事务传播机制配置错误** - **原因**:错误使用事务传播行为(如 `Propagation.NOT_SUPPORTED` 会挂起当前事务)。 - **示例**: ```java @Transactional(propagation = Propagation.NOT_SUPPORTED) public void methodC() { // 此方法不会在事务中运行 } ``` - **解决**:根据业务需求正确配置传播行为,如 `Propagation.REQUIRED`(默认)。 --- ### 4. **非 public 方法** - **原因**:SpringAOP 代理默认只对 `public` 方法生效,若在 `protected`/`private` 方法上使用 `@Transactional`,事务不会生效。 - **解决**:确保事务方法为 `public`。 --- ### 5. **数据库引擎不支持事务** - **原因**:如 MySQL 的 `MyISAM` 引擎不支持事务,而 `InnoDB` 支持。 - **解决**:检查并切换数据库引擎为支持事务的类型。 --- ### 6. **未启用事务管理** - **原因**:未在配置类中添加 `@EnableTransactionManagement`,或未配置事务管理器(如 `PlatformTransactionManager`)。 - **解决**:确保事务管理器已正确配置并启用。 --- ### 7. **多线程环境下事务隔离** - **原因**:新启动的线程中的数据库操作不在原事务上下文中。 - **解决**:避免在事务方法中开启新线程执行数据库操作。 --- ### 8. **异常被自定义切面(AOP)拦截后未抛出** - **原因**:如果其他切面(如日志切面)拦截了异常并处理,未继续抛出,事务无法感知异常。 - **解决**:确保异常最终能传递到事务管理器。 --- ### **排查建议** 1. **检查代理是否生效**:过调试查看对象是否是代理对象(如 `CGLIB$$` 结尾)。 2. **日志排查**:开启 Spring 事务日志(`logging.level.org.springframework.jdbc=DEBUG`)。 3. **确认配置**:检查 `@Transactional` 是否在实现类(而非接口)上,且事务管理器与数据源匹过检查以上场景,可以快速定位事务失效的原因并修复。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值