zoukankan      html  css  js  c++  java
  • SpringBoot AOP中JoinPoint的用法和通知切点表达式

    前言

    上一篇文章讲解了springboot aop 初步完整的使用和整合 这一篇讲解他的接口方法和类

    JoinPoint和ProceedingJoinPoint对象

    1. JoinPoint对象封装了SpringAop中切面方法的信息,在切面方法中添加JoinPoint参数,就可以获取到封装了该方法信息的JoinPoint对象.

    2. ProceedingJoinPoint对象是JoinPoint的子接口,该对象只用在@Around的切面方法中

    方法名 功能
    Signature getSignature(); 获取封装了署名信息的对象,在该对象中可以获取到目标方法名,所属类的Class等信息
    Object[] getArgs(); 获取传入目标方法的参数对象
    Object getTarget(); 获取被代理的对象
    Object getThis(); 获取代理对象
    @Aspect
    @Component
    public class aopAspect {
        /**
         * 定义一个切入点表达式,用来确定哪些类需要代理
         * execution(* aopdemo.*.*(..))代表aopdemo包下所有类的所有方法都会被代理
         */
        @Pointcut("execution(* aopdemo.*.*(..))")
        public void declareJoinPointerExpression() {}
    
        /**
         * 前置方法,在目标方法执行前执行
         * @param joinPoint 封装了代理方法信息的对象,若用不到则可以忽略不写
         */
        @Before("declareJoinPointerExpression()")
        public void beforeMethod(JoinPoint joinPoint){
            System.out.println("目标方法名为:" + joinPoint.getSignature().getName());
            System.out.println("目标方法所属类的简单类名:" +        joinPoint.getSignature().getDeclaringType().getSimpleName());
            System.out.println("目标方法所属类的类名:" + joinPoint.getSignature().getDeclaringTypeName());
            System.out.println("目标方法声明类型:" + Modifier.toString(joinPoint.getSignature().getModifiers()));
            //获取传入目标方法的参数
            Object[] args = joinPoint.getArgs();
            for (int i = 0; i < args.length; i++) {
                System.out.println("第" + (i+1) + "个参数为:" + args[i]);
            }
            System.out.println("被代理的对象:" + joinPoint.getTarget());
            System.out.println("代理对象自己:" + joinPoint.getThis());
        }
    
        /**
         * 环绕方法,可自定义目标方法执行的时机
         * @param pjd JoinPoint的子接口,添加了
         *            Object proceed() throws Throwable 执行目标方法
         *            Object proceed(Object[] var1) throws Throwable 传入的新的参数去执行目标方法
         *            两个方法
         * @return 此方法需要返回值,返回值视为目标方法的返回值
         */
        @Around("declareJoinPointerExpression()")
        public Object aroundMethod(ProceedingJoinPoint pjd){
            Object result = null;
    
            try {
                //前置通知
                System.out.println("目标方法执行前...");
                //执行目标方法
                //result = pjd.proeed();
                //用新的参数值执行目标方法
                result = pjd.proceed(new Object[]{"newSpring","newAop"});
                //返回通知
                System.out.println("目标方法返回结果后...");
            } catch (Throwable e) {
                //异常通知
                System.out.println("执行目标方法异常后...");
                throw new RuntimeException(e);
            }
            //后置通知
            System.out.println("目标方法执行后...");
    
            return result;
        }
    }
    

    切点表达式

    1. 在Spring AOP中,连接点始终代表方法的执行。切入点是与连接点匹配的,切入点表达语言是以编程方式描述切入点的方式。

    2. 切入点(Poincut)是定义了在“什么地方”进行切入,哪些连接点会得到通知。显然,切点一定是连接点

    3. 切点是通过@Pointcut注解和切点表达式定义的。@Pointcut注解可以在一个切面内定义可重用的切点。

    execute表达式

    *代表匹配任意修饰符及任意返回值,参数列表中..匹配任意数量的参数

    可以使用&&、||、!、三种运算符来组合切点表达式,表示与或非的关系

    1. 拦截任意公共方法execution(public * *(..))
    2. 拦截以set开头的任意方法execution(* set*(..))
    3. 拦截类或者接口中的方法
    拦截AccountService(类、接口)中定义的所有方法
    execution(* com.xyz.service.AccountService.*(..))
    
    1. 拦截包中定义的方法,不包含子包中的方法
    拦截com.xyz.service包中所有类中任意方法,**不包含**子包中的类
    execution(* com.xyz.service.*.*(..))
    
    1. 拦截包或者子包中定义的方法
    拦截com.xyz.service包或者子包中定义的所有方法
    execution(* com.xyz.service..*.*(..))
    

    通知分类

    @Before

    1. 前置通知: 在方法执行之前执行
    2. 前置通知使用@Before注解 将切入点表达式值作为注解的值

    @After

    1. 后置通知, 在方法执行之后执行
    2. 后置通知使用@After注解 ,在后置通知中,不能访问目标方法执行的结果

    @AfterRunning

    1. 返回通知, 在方法返回结果之后执行
    2. 返回通知使用@AfterRunning注解

    @AfterThrowing

    1. 异常通知, 在方法抛出异常之后执行
    2. 异常通知使用@AfterThrowing注解

    @Around

    1. 环绕通知, 围绕着方法执行
    2. 环绕通知使用@Around注解

    package com.jason.spring.aop.impl;
     
    import java.util.Arrays;
    import java.util.List;
     
    import org.aspectj.lang.JoinPoint;
    import org.aspectj.lang.ProceedingJoinPoint;
    import org.aspectj.lang.annotation.After;
    import org.aspectj.lang.annotation.AfterReturning;
    import org.aspectj.lang.annotation.AfterThrowing;
    import org.aspectj.lang.annotation.Around;
    import org.aspectj.lang.annotation.Aspect;
    import org.aspectj.lang.annotation.Before;
    import org.springframework.stereotype.Component;
     
     
    //把这个类声明为一个切面
    //1.需要将该类放入到IOC 容器中
    @Component
    //2.再声明为一个切面
    @Aspect
    public class LoggingAspect {
        
        //声明该方法是一个前置通知:在目标方法开始之前执行 哪些类,哪些方法
        //作用:@before 当调用目标方法,而目标方法与注解声明的方法相匹配的时候,aop框架会自动的为那个方法所在的类生成一个代理对象,在目标方法执行之前,执行注解的方法
        //支持通配符
        //@Before("execution(public int com.jason.spring.aop.impl.ArithmeticCaculatorImpl.*(int, int))")
        @Before("execution(* com.jason.spring.aop.impl.*.*(int, int))")
        public void beforeMethod(JoinPoint joinPoint){
            String methodName = joinPoint.getSignature().getName();
            List<Object> args = Arrays.asList(joinPoint.getArgs());
            System.out.println("The method " + methodName + " begins " + args);
        }
        
        /**
         * @Description:  在方法执行后执行的代码,无论该方法是否出现异常
         * @param joinPoint
         */
        @After("execution(* com.jason.spring.aop.impl.*.*(int, int))")
        public void afterMethod(JoinPoint joinPoint){
            String methodName = joinPoint.getSignature().getName();
            List<Object> args = Arrays.asList(joinPoint.getArgs());
            System.out.println("The method " + methodName + " end " + args);
        }
        
        /**
         * 
         * @Description:  在方法正常结束后执行代码,放回通知是可以访问到方法的返回值
         *
         * @param joinPoint
         */
        @AfterReturning( value="execution(* com.jason.spring.aop.impl.*.*(..))", returning="result")
        public void afterReturning(JoinPoint joinPoint ,Object result){
            String methodName = joinPoint.getSignature().getName();
            System.out.println("The method " + methodName + " end with " + result);
        }
        
        /**
         * 
         * @Description:  在目标方法出现异常时会执行代码,可以访问到异常对象,且,可以指定出现特定异常时执行通知代码
         *
         * @param joinPoint
         * @param ex
         */
        @AfterThrowing(value="execution(* com.jason.spring.aop.impl.*.*(..))",throwing="ex")
        public void afterThrowting(JoinPoint joinPoint, Exception  ex){
            String methodName = joinPoint.getSignature().getName();
            System.out.println("The method " + methodName + " occurs exceptions " + ex);
        }
        
        /**
         * 
         * @Description: 环绕通知需要携带 ProceedingJoinPoint 类型的参数
         *                    环绕通知 类似于  动态代理的全过程
         *                   ProceedingJoinPoint:可以决定是否执行目标方法
         *    环绕通知必须有返回值,返回值即为目标方法的返回值
         *    
         * @param proceedingJoinPoint
         */
        @Around("execution(* com.jason.spring.aop.impl.*.*(..))")
        public Object around(ProceedingJoinPoint proceedingJoinPoint){
            
            Object result = null;
            String methodName = proceedingJoinPoint.getSignature().getName();
            
            //执行目标方法
            try {
                //前置通知
                System.out.println("The method " + methodName + "begin with" + Arrays.asList(proceedingJoinPoint.getArgs()));
                
                result = proceedingJoinPoint.proceed();
                
                //后置通知
                System.out.println("The method " + methodName + "end with" + result);
                
            } catch (Throwable e) {
                //异常通知
                System.out.println("The method occurs exception : " + e);
                throw new RuntimeException();
            }
                //后置通知
                
            System.out.println("The method " + methodName + "end with" + result);
            
            return result;        
        }
    }
    

    切点表达式参考

  • 相关阅读:
    configparser模块
    xml文件解析
    shutil模块 + shelve模块 二合一版
    hashlib模块
    subprocess模块和sys模块
    re模块
    os模块
    random模块
    time模块、datetime模块讲解
    洛谷P3414 SAC#1
  • 原文地址:https://www.cnblogs.com/kenx/p/15093310.html
Copyright © 2011-2022 走看看