spring AOP 事务管理理解
怎样通过spring进行事务管理呢?下面进行简单的介绍~
假设我们现在有一个接口叫IBusinessService,里面有一个方法叫doSaveInformation(),有一个实现这个接口的类叫BusinessService,在doSaveInformation中有一系列的数据库操作,如下:
public interface IBusinessService{
public void doSaveInformation();
}
public class BusinessService implements IBusinessService{
public void doSaveInformation(){
进行一系列数据库操作;
}
}
我们想通过spring对这个方法进行事务管理,把这个方法当做一个事务去处理,该怎么做呢?
首先,在spring上下文中定义业务类:
<bean id="businessService" 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创建的代理类去调用该方法,而非套在其他的方法中。
欢迎大家指正~