事务与aop

事务配置

AOP的基本概念就不说了,其作用是为了将一组操作提取出来,在必要的时候再将其插入到代码中去。提取出来的一组操作就是一个aop切面组件,事务是aop最常用的场景之一,严格上说是声明式事务。

事务分为编程式事务和声明式事务。编程式事务指通过编码方式实现事务,管理使用TransactionTemplate,优点是灵活性更强,事务的粒度可以控制到语句级别。但缺点是对代码的侵入性较强。声明式事务是建立在aop之上的,其本质是对方法前后进行拦截,然后在目标方法开始之前创建或加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务只需要在配置好相关的事务规则即可,对代码的侵入性较低。声明式事务的最细粒度只能作用到方法级别,无法像编程式事务那样可以作用到代码块级别。

声明式事务又分为两种,一种是基于配置文件xml的,一种是基于注解的。

基于配置文件的事务需要在配置文件中定义事务的切面。先定义一个事务管理器(这里使用的是spring-jdbc的事务管理器):

id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
    <property name="dataSource" ref="dataSourceMaster"/>

然后配置事务通知,示例如下:

id="txAdvice" transaction-manager="transactionManager">
        
            name="get*" read-only="true" propagation="SUPPORTS" />
            name="query*" read-only="true" propagation="SUPPORTS" />
            name="delete*" propagation="REQUIRED" read-only="false"
                rollback-for="java.lang.Exception" />
            name="insert*" propagation="REQUIRED" read-only="false"
                rollback-for="java.lang.Exception" />
            name="update*" propagation="REQUIRED" read-only="false"
                rollback-for="java.lang.Exception" />
            name="save*" propagation="REQUIRED" read-only="false"
                rollback-for="java.lang.Exception" />
            name="*" isolation="DEFAULT" propagation="REQUIRED" />
        
    

基于注解的事务也需要先定义事务管理器,然后声明使用注解式事务。

<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 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());
        }
    }

} 
  

                            
                        
                    
                    
                    

你可能感兴趣的:(后端框架)