@Transactional修饰的方法的特点:public方法,且事务内尽可能简单
Spring事务是通过AOP实施事务增强的。由于接口的方法是public的,这句要求实现类的实现方法必须是public 的(不能是protect,private等),同时不能使用static的修饰符。即只有public的方法才可以被aop增强。
处理过程尽量的简单。尤其是带锁的事务方法,能不放在事务里面的最好不要放在事务里面。可以将常规的数据库查询操作放在事务前面进行,而事务内进行增、删、改、加锁查询等操作。
@Transactional与多数据源
@Transactional会和当前的数据源进行绑定,如果一个service中调用的dao使用不同的数据源则不能切换。其默认使用的是配置文件中最先置顶的事务管理器。
可通过@Transactional(transactionManager = "dataSourceTxManagerB")的方式置顶数据源;
或者使用aop切面在进入对应的service层方法之前,切换到对应的数据源的方式来处理。即使用ThreadLocal来存储当前线程的数据源。
Aop切面导致的事务传播机制失效:Aop切面实质调用的是对应的代理对象
问题:在一个service方法A的内部,通过this调用该service的内部其他方法B时,会导致方法B的事务传播机制失效。但是在A的内部调用其他service的方法B时,B的事务传播机制生效。如下程序调用,insertUserLog()的事务传播机制将失效。insert(){userDao.insertUser();this.insertUserLog();}
原理:在对AServiceImpl的某个方法添加注解@Transactional时,spring会自动给该类生成对应的代理类AProxyServiceImpl。而AOP的事务操作以及前置后置增强均是建立在搭理类上的。
即调用逻辑为外部-->A'-->A方法1-->A方法2-->A'-->外部。