事务配置
AOP的基本概念就不说了,其作用是为了将一组操作提取出来,在必要的时候再将其插入到代码中去。提取出来的一组操作就是一个aop切面组件,事务是aop最常用的场景之一,严格上说是声明式事务。
事务分为编程式事务和声明式事务。编程式事务指通过编码方式实现事务,管理使用TransactionTemplate,优点是灵活性更强,事务的粒度可以控制到语句级别。但缺点是对代码的侵入性较强。声明式事务是建立在aop之上的,其本质是对方法前后进行拦截,然后在目标方法开始之前创建或加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务只需要在配置好相关的事务规则即可,对代码的侵入性较低。声明式事务的最细粒度只能作用到方法级别,无法像编程式事务那样可以作用到代码块级别。
声明式事务又分为两种,一种是基于配置文件xml的,一种是基于注解的。
基于配置文件的事务需要在配置文件中定义事务的切面。先定义一个事务管理器(这里使用的是spring-jdbc的事务管理器):
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSourceMaster"/>
</bean>
然后配置事务通知,示例如下:
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="get*" read-only="true" propagation="SUPPORTS" />
<tx:method name="query*" read-only="true" propagation="SUPPORTS" />
<tx:method name="delete*" propagation="REQUIRED" read-only="false"
rollback-for="java.lang.Exception" />
<tx:method name="insert*" propagation="REQUIRED" read-only="false"
rollback-for="java.lang.Exception" />
<tx:method name="update*" propagation="REQUIRED" read-only="false"
rollback-for="java.lang.Exception" />
<tx:method name="save*" propagation="REQUIRED" read-only="false"
rollback-for="java.lang.Exception" />
<tx:method name="*" isolation="DEFAULT" propagation="REQUIRED" />
</tx:attributes>
</tx:advice>
基于注解的事务也需要先定义事务管理器,然后声明使用注解式事务。
<tx:annotation-driven transaction-manager="transactionManager"/>
配置好之后只需要在需要开启事务的方法上加上@Transactional({事务传播方式})即可。
AOP配置方式
同样aop也有两种配置方式,分别是基于xml的方式和基于注解的方式。基于xml配置的aop需要定义好事务的切面类,在该类中定义aop的通知方法。然后在配置文件中将其作为一个切面bean使用:
<bean id="beanAspect" class="xx.framework.validator.BeanAspect"/>
<aop:config>
<aop:pointcut id="valid" expression="@annotation(com.xx.framework.interceptor.validation.BeanValid)"/>
<aop:aspect id="validAspect" ref="beanAspect" order="1">
<aop:before method="checkBeanValid" pointcut-ref="valid"/>
</aop:aspect>
</aop:config>
注解式事务只需要加上Spring组件注解@Compont和切面注解@Aspect,然后在通知方法上加上切入点注解即可。如下所示:
@Component
@Aspect
public class BeanAspect {
@Before("@annotation(com.tuniu.xff.drip.framework.interceptor.validation.BeanValid)")
public void checkBeanValid(JoinPoint joinPoint) throws Exception{
List<Object> args = Arrays.asList(joinPoint.getArgs());
try {
args.stream().forEach((Object obj) -> {
DataValidationUtil.validate(obj);
});
} catch (ValidException e) {
throw new BusinessException(ErrorCodeDefinition.INVALID_PARAMETER, "参数不合法," + e.getMessage());
}
}
}