关于spring事务的几个误区记录

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(是否暴露代理对象),否则获取不到当前代理对象

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容