• (续)SSM整合之spring笔记(AOP 基于注解的AOP切点表达式的语法,获取连接点的信息, 重用切入点表达式)(P102)


    一  .切入点表达式语法

    ①作用

    ②语法细节

    * 号代替 权限修饰符 返回值 部分表示 权限修饰符 返回值 不限
    在包名的部分,一个 “*” 号只能代表包的层次结构中的一层,表示这一层是任意的。
    例如: *.Hello 匹配 com.Hello ,不匹配 com.atguigu.Hello
    在包名的部分,使用 “*..” 表示包名任意、包的层次深度任意
    在类名的部分,类名部分整体用 * 号代替,表示类名任意
    在类名的部分,可以使用 * 号代替类名的一部分
    例如: *Service 匹配所有名称以 Service 结尾的类或接口
    在方法名部分,可以使用 * 号表示方法名任意
    在方法名部分,可以使用 * 号代替方法名的一部分
    例如: *Operation 匹配所有方法名以 Operation 结尾的方法
    在方法参数列表部分,使用 (..) 表示参数列表任意
    在方法参数列表部分,使用 (int,..) 表示参数列表以一个 int 类型的参数开头
    在方法参数列表部分,基本数据类型和对应的包装类型是不一样的
    切入点表达式中使用 int 和实际方法中 Integer 是不匹配的
    在方法返回值部分,如果想要明确指定一个返回值类型,那么必须同时写明权限修饰符
    例如: execution(public int .. Service.*(.., int)) 正确
    例如: execution(* int .. Service.*(.., int)) 错误

     ③ 具体用法

    切点表达式

    1. /*1、在切面中,需要通过指定的注解将方法标识为通知方法
    2. * @Before:前置通知,在目标对象方法执行之前执行
    3. *2、切入点表达式:设置在标识通知的注解的value属性中
    4. * execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int)
    5. * execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..)
    6. * 第一个*表示任意的访问修饰符和返回值类型
    7. * 第二个*表示类中任意的方法
    8. * ..表示任意的参数列表
    9. * 类的地方也可以使用*,表示包下所有的类
    10. * */
    11. @Component
    12. @Aspect //将当前组件标识为切面
    13. public class LoggerAspect {
    14. @Before("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    15. //@Before("execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int))")
    16. public void beforeAdviceMethod() {
    17. System.out.println("LoggerAspect,前置通知");
    18. }
    19. }

    测试:

    1. public class AOPByAnnotationTest {
    2. @Test
    3. public void testAOPByAnnotation(){
    4. //获取IOC容器
    5. ApplicationContext ioc = new ClassPathXmlApplicationContext("aop-annotation.xml");
    6. //获取目标对象
    7. // CalculatorImpl calculator = ioc.getBean(CalculatorImpl.class);
    8. //获取代理对象
    9. Calculator calculator = ioc.getBean(Calculator.class);
    10. calculator.sub(10, 1);
    11. }
    12. }

     

     

    获取连接点信息

    1. /*1、在切面中,需要通过指定的注解将方法标识为通知方法
    2. * @Before:前置通知,在目标对象方法执行之前执行
    3. *2、切入点表达式:设置在标识通知的注解的value属性中
    4. * execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int)
    5. * execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..)
    6. * 第一个*表示任意的访问修饰符和返回值类型
    7. * 第二个*表示类中任意的方法
    8. * ..表示任意的参数列表
    9. * 类的地方也可以使用*,表示包下所有的类
    10. * */
    11. @Component
    12. @Aspect //将当前组件标识为切面
    13. public class LoggerAspect {
    14. @Before("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    15. //@Before("execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int))")
    16. public void beforeAdviceMethod() {
    17. System.out.println("LoggerAspect,前置通知");
    18. }
    19. }

    我们之前在动态代理里面   我们在前置通知的位置  也就是在目标对象的方法执行之前

    System.out.println("LoggerAspect,前置通知");输出的是调用方法的方法名  和参数列表 

    用了前置通知之后   不知道应该如何获取  还没有动态代理实现的功能多 我们连连接点的信息都获取不到  也就是说我们要加入的通知的方法的一些信息 我们都获取不到

    那我们获取到呢

    加入JoinPoint joinPoint   帮助我们获取连接点信息   换句话说 切点表达式定位的是哪个方法  那我们JoinPoint joinPoint 表示的是哪个方法的信息   

    1. package com.atguigu.spring.aop.annotation;
    2. import org.aspectj.lang.JoinPoint;
    3. import org.aspectj.lang.Signature;
    4. import org.aspectj.lang.annotation.Aspect;
    5. import org.aspectj.lang.annotation.Before;
    6. import org.springframework.stereotype.Component;
    7. import java.util.Arrays;
    8. /**
    9. * 1、在切面中,需要通过指定的注解将方法标识为通知方法
    10. * @Before:前置通知,在目标对象方法执行之前执行
    11. *
    12. * 2、切入点表达式:设置在标识通知的注解的value属性中
    13. * execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int)
    14. * execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..)
    15. * 第一个*表示任意的访问修饰符和返回值类型
    16. * 第二个*表示类中任意的方法
    17. * ..表示任意的参数列表
    18. * 类的地方也可以使用*,表示包下所有的类
    19. *
    20. * 3、获取连接点的信息
    21. * 在通知方法的参数位置,设置JoinPoint类型的参数,就可以获取连接点所对应方法的信息
    22. * //获取连接点所对应方法的签名信息
    23. * Signature signature = joinPoint.getSignature();
    24. * //获取连接点所对应方法的参数
    25. * Object[] args = joinPoint.getArgs();
    26. */
    27. @Component
    28. @Aspect //将当前组件标识为切面
    29. public class LoggerAspect {
    30. @Before("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    31. //@Before("execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int))")
    32. public void beforeAdviceMethod(JoinPoint joinPoint) {
    33. //获取连接点所对应方法的签名信息
    34. Signature signature = joinPoint.getSignature();
    35. //获取连接点所对应方法的参数
    36. Object[] args = joinPoint.getArgs();
    37. System.out.println("LoggerAspect,方法:"+signature.getName()+",参数:"+ Arrays.toString(args));
    38. }
    39. }

    重用切入点表达式

    1. /**
    2. * Date:2022/7/4
    3. * Author:ybc
    4. * Description:
    5. * 1、在切面中,需要通过指定的注解将方法标识为通知方法
    6. * @Before:前置通知,在目标对象方法执行之前执行
    7. * @After:后置通知,在目标对象方法的finally字句中执行
    8. *
    9. * 2、切入点表达式:设置在标识通知的注解的value属性中
    10. * execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int)
    11. * execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..)
    12. * 第一个*表示任意的访问修饰符和返回值类型
    13. * 第二个*表示类中任意的方法
    14. * ..表示任意的参数列表
    15. * 类的地方也可以使用*,表示包下所有的类
    16. * 3、重用切入点表达式
    17. * //@Pointcut声明一个公共的切入点表达式
    18. * @Pointcut("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    19. * public void pointCut(){}
    20. * 使用方式:@Before("pointCut()")
    21. *
    22. * 4、获取连接点的信息
    23. * 在通知方法的参数位置,设置JoinPoint类型的参数,就可以获取连接点所对应方法的信息
    24. * //获取连接点所对应方法的签名信息
    25. * Signature signature = joinPoint.getSignature();
    26. * //获取连接点所对应方法的参数
    27. * Object[] args = joinPoint.getArgs();
    28. *
    29. */
    30. @Component
    31. @Aspect //将当前组件标识为切面
    32. public class LoggerAspect {
    33. @Pointcut("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    34. public void pointCut(){}
    35. //@Before("execution(public int com.atguigu.spring.aop.annotation.CalculatorImpl.add(int, int))")
    36. //@Before("execution(* com.atguigu.spring.aop.annotation.CalculatorImpl.*(..))")
    37. @Before("pointCut()")
    38. public void beforeAdviceMethod(JoinPoint joinPoint) {
    39. //获取连接点所对应方法的签名信息
    40. Signature signature = joinPoint.getSignature();
    41. //获取连接点所对应方法的参数
    42. Object[] args = joinPoint.getArgs();
    43. System.out.println("LoggerAspect,方法:"+signature.getName()+",参数:"+ Arrays.toString(args));
    44. }
    45. @After("pointCut()")
    46. public void afterAdviceMethod(){
    47. }
    48. }

     

     
    

     
    

  • 相关阅读:
    传言称 iPhone 16 Pro 将支持 40W 快速充电和 20W MagSafe
    【企业安全】企业安全系列第 2 部分 — 身份和访问管理
    反射 - 枚举 - Lambda表达式
    spring boot 分布式session实现
    Java反射原理和实际用法
    spring6-事务
    供应商合同管理失败的原因有哪些?
    重看工厂模式
    在ubuntu上安装hadoop完分布式
    景区数字化营销怎么做?景区数字化营销优势
  • 原文地址:https://blog.csdn.net/m0_59281987/article/details/127723364