欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

浅谈Spring的两种事务定义方式

程序员文章站 2024-02-22 10:04:47
一、声明式 这种方法不需要对原有的业务做任何修改,通过在xml文件中定义需要拦截方法的匹配即可完成配置,要求是,业务处理中的方法的命名要有规律,比如setxxx,xxxu...

一、声明式

这种方法不需要对原有的业务做任何修改,通过在xml文件中定义需要拦截方法的匹配即可完成配置,要求是,业务处理中的方法的命名要有规律,比如setxxx,xxxupdate等等。详细配置如下:

<bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager"> 
  <property name="datasource" ref="datasource"></property> 
 </bean> 
  
 <!-- 定义事务通知 --> 
 <tx:advice id="txadvice" transaction-manager="transactionmanager"> 
  <!-- 定义方法的过滤规则 --> 
  <tx:attributes> 
   <!-- 所有方法都使用事务 --> 
   <tx:method name="*" propagation="required"/> 
   <!-- 定义所有get开头的方法都是只读的 --> 
   <tx:method name="get*" read-only="true"/> 
  </tx:attributes> 
 </tx:advice> 
  
 <!-- 定义aop配置 --> 
 <aop:config> 
  <!-- 定义一个切入点 --> 
  <aop:pointcut expression="execution (* com.test.services.impl.*.*(..))" id="services"/> 
  <!-- 对切入点和事务的通知,进行适配 --> 
  <aop:advisor advice-ref="txadvice" pointcut-ref="services"/> 
 </aop:config> 

二、注解式

这种方法,只需要在spring配置文件中定义一个事务管理对象(如datasourcetransactionmanager),然后加入<tx:annotation-driven/>节点,引用该事务管理对象,然后即可在需要进行事务处理的类和方法使用@transactional进行标注。示例如下:

<bean id="transactionmanager" class="org.springframework.jdbc.datasource.datasourcetransactionmanager"> 
  <property name="datasource" ref="datasource"></property> 
 </bean> 
 <!-- 声明使用注解式事务 --> 
 <tx:annotation-driven transaction-manager="transactionmanager"/> 

三、<tx:annotation-driven/>

spring提供的<tx:annotation-driven/>用于开启对注解事务管理的支持,从而能识别bean类上的@transactional注解元数据,其具有以下属性:

1、<tx:annotation-driven/>

transaction-manager:指定事务管理器名字,默认为transactionmanager,当使用其他名字时需要明确指定;

proxy-target-class:表示将使用的代码机制,默认false表示使用jdk代理,如果为true将使用cglib代理

order:定义事务通知顺序,默认ordered.lowest_precedence,表示将顺序决定权交给aop来处理。

2、@transaction

spring使用@transaction来指定事务属性,可以在接口、类或方法上指定,如果类和方法上都指定了@transaction,则方法上的事务属性被优先使用,具体属性如下:

value:指定事务管理器名字,默认使用<tx:annotation-driven/>指定的事务管理器,用于支持多事务管理器环境;

propagation:指定事务传播行为,默认为required,使用propagation.required指定;

isolation:指定事务隔离级别,默认为“default”,使用isolation.default指定;

readonly:指定事务是否只读,默认false表示事务非只读;

timeout:指定事务超时时间,以秒为单位,默认-1表示事务超时将依赖于底层事务系统;

rollbackfor:指定一组异常类,遇到该类异常将回滚事务;

rollbackforclassname:指定一组异常类名字,其含义与<tx:method>中的rollback-for属性语义完全一样;

norollbackfor:指定一组异常类,即使遇到该类异常也将提交事务,即不回滚事务;

norollbackforclassname:指定一组异常类名字,其含义与<tx:method>中的no-rollback-for属性语义完全一样;

spring提供的@transaction注解事务管理内部同样利用环绕通知transactioninterceptor实现事务的开启及关闭。

使用@transactional注解事务管理需要特别注意以下几点:

如果在接口、实现类或方法上都指定了@transactional 注解,则优先级顺序为方法>实现类>接口;

建议只在实现类或实现类的方法上使用@transactional,而不要在接口上使用,这是因为如果使用jdk代理机制是没问题,因为其使用基于接口的代理;而使用使用cglib代理机制时就会遇到问题,因为其使用基于类的代理而不是接口,这是因为接口上的@transactional注解是“不能继承的”;

在spring代理机制下(不管是jdk动态代理还是cglib代理),“自我调用”同样不会应用相应的事务属性,其语义和<tx:tags>中一样;

默认只对runtimeexception异常回滚;

在使用spring代理时,默认只有在public可见度的方法的@transactional 注解才是有效的,其它可见度(protected、private、包可见)的方法上即使有@transactional 注解也不会应用这些事务属性的,spring也不会报错,如果你非要使用非公共方法注解事务管理的话,可考虑使用aspectj。

以上这篇浅谈spring的两种事务定义方式就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持。