您的位置:首页 > 编程语言 > Java开发

spring注解事务

2016-07-28 00:00 489 查看
摘要: spring @Transactional

假设我们现在有一个接口叫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" />

其次,为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通过接口方式实现的代理模式

这里做一个浅显的解释,在通过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创建的代理类去调用该方法,而非套在其他的方法中。

转载:
http://blog.knowsky.com/190629.htm
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: