Spring 学习之路(九):Spring 中的AOP(二):事务通知

AspectJ
  • 目前,spring 框架中我们可以使用基于 AspectJ 注解或者是基于XML配置的 AOP(主流是使用 AspectJ ,简单,方便)
  • 如何配置AspectJ

    1. 简单理解,AspectJ 就是一个支持 aop 的第三方组件,spring 提供了很好的支持,我们只需要将 对应的 jar 包加入我们的项目即可(对应 jar 包可以在我的源代码下载)
    2. 如图:
      mark

    3. 配置文件中声明 使用 AspectJ 注解

      1. 引入aop命名空间
        mark
      2. 使 AspectJ 注解生效
        mark
  • AspectJ 注解工作流程

    1. 在 Spring 中声明 AspectJ 切面, 只需要在 IOC 容器中将切面声明为 Bean 实例. 当在 Spring IOC 容器中初始化 AspectJ 切面之后, Spring IOC 容器就会为那些与 AspectJ 切面相匹配的 Bean 创建代理
    2. 在 AspectJ 注解中, 切面只是一个带有 @Aspect 注解的 Java 类

    3. 通知是标注有某种注解的简单的 Java 方法

    4. 对以上流程不清楚的话,我们直接看代码
前置通知
//把该类声明为一个切面:需要把该类放入到ioc容器中,然后再声明为一个切面
@Aspect
@Component
public class LogginAspect {

    // 声明该方法是一个前置通知,在目标方法开始之前执行
    @Before("execution(void com.zc.cris.beans.spring.aop.impl.Chinese.*(String))")
    public void beforeMethod(JoinPoint joinPoint) {
        // 获取方法签名和参数集合
        System.out.println(joinPoint.getSignature().getName() + "-----" + Arrays.asList(joinPoint.getArgs()));
        System.out.println("我是方法的前置通知");
    }

- 测试代码

    @Test
    void testProxy() {
        //创建ioc容器
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

        //获取bean
        People bean = context.getBean(People.class);

//      System.out.println(bean); 
//      System.out.println(bean.getClass().getName());

        //System.out.println(bean instanceof Chinese);      //false

        //使用bean
        bean.eat("筷子");
        bean.say("中文");
    }

console:

mark

后置通知
// 声明该方法是一个后置通知,在目标方法执行后执行(无论目标方法是否发生异常)
    // 且后置通知无法访问目标方法的返回值
    @After("execution(* com.zc.cris.beans.spring.aop.impl.*.*(String))")
    public void afterMethod(JoinPoint joinPoint) {

        System.out.println("我是方法的后置通知");
    }

- 测试代码同上

console:

mark

返回通知
    // 声明该方法为返回通知:方法正常执行结束后执行的代码
    // 返回 通知是可以访问到方法的返回值的!
    /*切点表达式表示执行任意类的任意方法. 第
    一个 * 代表匹配任意修饰符及任意返回值,  第二个 * 代表任意类的对象,
    第三个 * 代表任意方法, 参数列表中的 ..  匹配任意数量的参数
    */
    @AfterReturning(value = "execution(public void com.zc.cris.beans.spring.aop.impl.*.*(..))", returning = "result")
    public void afterRetruning(JoinPoint joinPoint, Object result) {

        System.out.println("我是方法的返回通知" + joinPoint.getSignature().getName() + "^^^^"
                + Arrays.asList(joinPoint.getArgs() + "我是方法的返回值" + result));
    }


- 测试代码同上

console:

mark

异常通知
//目标方法出现异常才会指定的代码
    //可以访问到异常对象,且可以指定出现特定的异常(NullPointException)才会执行
    @AfterThrowing(value = "execution(public void com.zc.cris.beans.spring.aop.impl.*.*(..))",
            throwing="e")
    public void afterThrowing(JoinPoint joinPoint, Exception e) {
        System.out.println("我是目标方法发生异常才执行的通知:"+e.getMessage());
    }

- 测试代码同上

console:

mark

  • 通过以上四种通知类型的应用我们大致了解Spring 的aop通过 AspectJ 组件是如何完成的,让我们再 修改之前的代理类,加深理解

mark

环绕通知
    //环绕通知:必须携带 ProceedingJoinPoint 类型的参数
    //环绕通知类似于动态代理的全过程:ProceedingJoinPoint 类型的参数可以决定目标方法的执行,
    //环绕通知必须要有返回值,返回值其实就是目标方法的返回值
    @Around(value = "execution(public void com.zc.cris.beans.spring.aop.impl.*.*(..))")
    public Object around(ProceedingJoinPoint pjt) {

        Object result = null;
        String methodName = pjt.getSignature().getName();

        try {
            //前置通知
            System.out.println("我是环绕通知的前置通知!!!!!");

            //执行目标方法
            result = pjt.proceed();

            //返回通知
            System.out.println("我是环绕通知的后置通知");
        } catch (Throwable e) {
            //异常通知
            System.out.println("我是环绕通知的异常通知"+e.getMessage());
            throw new RuntimeException(e);
        }
        //后置通知
        System.out.println("我是环绕通知的后置通知");
        return result;
    }

console:

mark

切面的优先级
  • 假如我们现在有两个切面类,一个负责参数验证,一个负责日志记录,那么我们如何确定这两个切面类谁先执行,谁后执行呢?
//使用 @Order(1) 注解指定切面的优先级,数字越小,优先级越高
@Order(1)
@Aspect
@Component
public class ValidationAspect {

    @Before(value = "execution(* com.zc.cris.beans.spring.aop.impl.*.*(..))")
    public void validate(JoinPoint joinPoint) {

        System.out.println("------- validation-----"+ Arrays.asList(joinPoint.getArgs()));
    }
}

@Order(2)
//把该类声明为一个切面:需要把该类放入到ioc容器中,然后再声明为一个切面
@Aspect
@Component
public class LogginAspect {

console:

mark

切面表达式的重用
  • 通过上面的测试,我们发现每个通知的注解里都需要写相同的切面表达式,这明显不符合我们的风格,著名编程大师马丁·富勒 就曾经说过,代码有很多种坏味道,而重复是最坏的一种,事实上通过一个小小的注解就可以搞定
    /*
     * 定义一个方法,专门用来声明切入点表达式,一般的,该方法中不需要再写任何代码
     * 使用@Pointcut 注解来声明
     * 后面的其他通知直接使用该方法名来引用当前的切入点表达式即可
     */
    @Pointcut("execution(* com.zc.cris.beans.spring.aop.impl.*.*(String))")
    public void declaredJointPointExpresson() {};


    // 声明该方法是一个前置通知,在目标方法开始之前执行
    @Before("declaredJointPointExpresson()")


    @Before(value = "com.zc.cris.beans.spring.aop.impl.LogginAspect.declaredJointPointExpresson()")
  • 我们通过@Pointcut 注解对切面表达式进行了重构,当前类的通知或者其他包的类的通知,都可以使用,以达到简洁,高效的目的
通过xml配置文件来配置spring的事务通知(不推荐,看完代码你就知道为什么了,了解即可)
  • applicationContext.aopXML.xml
<bean id="chinese" class="com.zc.cris.beans.spring.aop.impl.xml.Chinese"></bean>

    <bean id="validationAspect" class="com.zc.cris.beans.spring.aop.impl.xml.ValidationAspect"></bean>

    <bean id="logginAspect" class="com.zc.cris.beans.spring.aop.impl.xml.LogginAspect"></bean>

    <!-- 配置aop -->
    <aop:config>
        <!-- 定义切入点表达式 -->
        <aop:pointcut expression="execution(* com.zc.cris.beans.spring.aop.impl.xml.*.*(String))" id="pointCut"/>
        <!-- 定义一个切面对象 -->
        <aop:aspect ref="logginAspect" order="2">
            <!-- 定义各种通知 -->
            <aop:before method="beforeMethod" pointcut-ref="pointCut"/>
            <aop:after method="afterMethod" pointcut-ref="pointCut"/>
            <aop:after-returning method="afterRetruning" returning="result" pointcut-ref="pointCut"/>
            <aop:after-throwing method="afterThrowing" throwing="e" pointcut-ref="pointCut"/>
            <!-- <aop:around method="around"/> -->
        </aop:aspect>
        <aop:aspect ref="validationAspect" order="1">
            <aop:before method="validate" pointcut-ref="pointCut"/>
        </aop:aspect>       
    </aop:config>
  • 取消我们aop切面类上的所有切面注解,然后进行测试发现console打印的和之前测试一毛一样

  • 源代码点我

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值