spring的事务管理是平时常用的一个点,但是以前利用的时候有很多误区,这里特意记录一下,希望以后引起警戒
1,方法修饰符问题:
private 修饰的方法没有事务,因为代理类无法为私有的方法提供扩展,至于为什么私有的方法不能被代理,简单点来说,spring的动态代理一般有两种,基于接口的JDK InvocationHandler 和基于继承的cgLib,无论是基于接口和继承,私有的属性方法本身就不能被其他类去访问,所以就自然不能进行代理了(ps:主要是太深了我也就不太懂了)
/**
* 先添加后删除,删除的逻辑另起事务
* @param order
*/
public void testInsert(BusinessOrder order){
businessOrderMapper.insert(order);
testDeleteNewTx(1L);
int i = 1 / 0;
}
/**
* 这里会new新的事务
* @param wayBillId
*/
@Transactional(propagation=Propagation.REQUIRES_NEW)
private void testDeleteNewTx(Long wayBillId){
Example example = new Example(BusinessOrder.class,false,false);
Example.Criteria criteria = example.createCriteria();
criteria.andEqualTo("wayBillId",wayBillId);
businessOrderMapper.deleteByExample(example);
}
这里我们的逻辑是先添加,后删除,而且删除逻辑不受外部事务的回滚的影响,这里是自己配置了PROPAGATION_REQUIRES_NEW的隔离级别,但是,因为方法是私有的所以事务不生效,外面的事务依旧会影响testDeleteNewTx的结果,那么我们改进一下?将方法修饰符改成public的就可以了吗?
/**
* 先添加后删除,删除的逻辑另起事务
* @param order
*/
@Transactional
public void testInsert(BusinessOrder order){
businessOrderMapper.insert(order);
testDeleteNewTx(1L);
int i = 1 / 0;
}
/**
* 这里会new新的事务
* @param wayBillId
*/
@Transactional
public void testDeleteNewTx(Long wayBillId){
Example example = new Example(BusinessOrder.class,false,false);
Example.Criteria criteria = example.createCriteria();
criteria.andEqualTo("wayBillId",wayBillId);
businessOrderMapper.deleteByExample(example);
}
这么修改后其实还是不对的,因为,spring的事务依赖代理类,这里的testDeleteNewTx是this对象调用,this对象并不是一个代理类,所有还是不生效的,这也是日常开发中经常犯得错误,改成如下:
@Autowired
private BusinessOrderServiceImpl businessOrderService;
/**
* 先添加后删除,删除的逻辑另起事务
* @param order
*/
public void testInsert(BusinessOrder order){
businessOrderMapper.insert(order);
// //1,利用spring自动注入一个代理类:BusinessOrderServiceImpl
// businessOrderService.testDeleteNewTx(1L);
//2,利用AopContext获取当前代理对象,注意这里需要将 expose-proxy设置成true
BusinessOrderServiceImpl proxy = (BusinessOrderServiceImpl)AopContext.currentProxy();
proxy.testDeleteNewTx(1L);
int i = 1 / 0;
}
/**
* 这里会new新的事务
* @param wayBillId
*/
public void testDeleteNewTx(Long wayBillId){
Example example = new Example(BusinessOrder.class,false,false);
Example.Criteria criteria = example.createCriteria();
criteria.andEqualTo("wayBillId",wayBillId);
businessOrderMapper.deleteByExample(example);
}
本类中调用主要有两种解决方案:
1,利用spring自动注入一个本类的代理类
2,利用AopContext获取当前代理对象,注意这里需要将 expose-proxy设置成true(是否暴露代理对象),否则获取不到当前代理对象