Spring基于常用AspectJ切点表达式使用介绍

 更新时间:2022年12月20日 10:24:08   作者:码畜c  
AspectJ是一个基于Java语言的AOP框架,使用AspectJ需要导入Spring AOP和AspectJ相关jar包,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

execution (常用,方法级别的匹配)

语法:

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern) throws-pattern?)

  • modifiers-pattern:方法的访问符,如public、protected、default,不能匹配private方法
  • ret-type-pattern:方法的返回值类型,例:java.util.List、java.lang.String(类的全限定名,或者支持的简写如String、Integer)
  • declaring-type-pattern:方法所在类的全路径名,例:life.cqq.controller.UserController
  • name-pattern:方法名
  • param-pattern:方法的参数类型,例:java.util.List、java.lang.String
  • throws-pattern:方法抛出的异常类型,例:java.lang.RuntimeException、java.lang.Exception

(注:返回值类型 & 参数类型支持泛型判断)

带有?的位,非必写,即可省略。所以,一个表达式至少由3部分组成。例:

// 匹配所有方法
execution(* *(..))

若该位省略,例modifiers-pattern,意味着匹配该位能匹配的全部方法:public、protected、default修饰的方法

上例子:

@Pointcut("execution(public java.util.List<java.lang.Integer> *.test(java.util.List<java.lang.String>) throws java.lang.RuntimeException)")

匹配所有的方法名为test,方法格式为下方格式的方法。

public List<Iteger> test(List<String> list) throws RuntimeException

通配符

  • “..” : 应用在declaring-type-pattern上时,意味着扫描子孙包。应用在param-pattern时意味着任意类型
  • “*” :任意匹配符
  • “+” :表示按照类型匹配,必须追加在declaring-type-pattern中的类名后,表示所有的该类型的类、继承和扩展该类型的类

上例子:

// 1. 关于 .. 通配符
// 匹配子孙包下的所有类的所有方法
execution(* life.cqq..*.*(..))

// 2. 关于 * 通配符
// 匹配所有类的所有方法
execution(* *(..))

// 匹配子孙包下带有指定前缀的包下的所有类的所有方法
execution(* life.cqq..prefix*.*.*(..))

// 3. 关于 + 通配符
// 匹配子孙包下继承或实现了指定类型的类 以及 该类型的本类(若该类型不为接口) 下的所有方法
execution(* life.cqq..type+.*(..))

within (常用,类级别的匹配) 语法:

within(declaring-type-pattern)

execution语法中的declaring-type-pattern部分,从通配符的例子中提取出可应用于within的内容:

// 1. 关于 .. 通配符
// 匹配子孙包下的所有类的所有方法
within(life.cqq..*)

// 2. 关于 * 通配符
// 匹配所有类的所有方法
within(*)

// 匹配子孙包下带有指定前缀的包下的所有类的所有方法
within(life.cqq..prefix*.*)

// 3. 关于 + 通配符
// 匹配子孙包下继承或实现了指定类型的类 以及 该类型的本类(若该类型不为接口) 下的所有方法
within(life.cqq..type+)

@annotation (常用,方法级别的匹配)

语法:

@annotation(annotation-type-pattern)

与execution一样针对于方法,匹配添加了指定注解的方法。

@within (常用,类级别的匹配)

语法:

@within(annotation-type-pattern)

与within一样针对于类,匹配添加了指定注解的类。

逻辑运算符

  • &&
  • ||
  • !

应用在多个Point上组成稍微复杂的匹配表达式

上例子:

@Component
class AopBean {
    public void test(String str) {
        System.out.println("String");
    }
    public void test(Integer integer) {
        System.out.println("Integer");
    }
}
@Pointcut("within(life.cqq.aop.logicsymbol.AopBean)")
public void point1() {}
@Pointcut("execution(public void test(java.lang.String))")
public void point2() {}
@Pointcut("execution(public void test(java.lang.Integer))")
public void point3() {}
// @Around("point1() && point2()")               : 匹配参数为String类型的方法
// @Around("point1() && (point2() || point3())") : 匹配参数为Integer、String类型的方法
// @Around("point1() && !point2())")             : 匹配参数为Integer类型的方法

以上仅为平时常用的内容,还有其他许多写法,如:args、@args、target、@target等

一次实际应用

需求:

  • 基于AOP + 自定义注解的方式实现三个系统的接口请求参数的记录。
  • 需实现就近原则:优先判断方法上日志注解,若无在判断方法上的注解。类上添加日志注解时,意味着类的全部方法都需要打印请求参数。
@Pointcut("execution(* life.cqq..controller.*.*(..))")
private void log() {}
@Pointcut("@within(life.cqq.common.newlog.annotation.AppOpnLog) || @annotation(life.cqq.common.newlog.annotation.AppOpnLog)")
public void appOpnLog() {
}
@Pointcut("@within(life.cqq.common.newlog.annotation.EsOpnLog) || @annotation(life.cqq.common.newlog.annotation.EsOpnLog)")
public void esOpnLog() {
}
@Pointcut("@within(life.cqq.common.newlog.annotation.WebOpnLog) || @annotation(life.cqq.common.newlog.annotation.WebOpnLog)")
public void webOpnLog() {
}
@Around("appOpnLog() || esOpnLog() || webOpnLog()")
public Object opnLogAround(ProceedingJoinPoint point) throws Throwable {
    // ......
    return point.proceed();
}
  • 因为是三个系统,log切点表达式中的controller前需要使用通配符"…"。因为每个系统的controller包名都是不一致的,但可以确定都符合格式: life.cqq.xxx.controller。
  • 基于注解的形式,且实现就近原则,那么既要匹配类上的日志注解也要匹配方法上的日志注解,故使用@within 与 @annotation 并用逻辑或连接。具体是在方法还是在类上添加的注解,在Around方法中解析处理。

到此这篇关于Spring基于常用AspectJ切点表达式使用介绍的文章就介绍到这了,更多相关Spring AspectJ内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • SpringBoot内置数据源的持久化与解决方案

    SpringBoot内置数据源的持久化与解决方案

    数据源的配置 我们先基于SpringBoot默认的HikariDataSource数据源,导入JDBC场景,看看SpringBoot帮我们自动配置了什么,下面我们来了解SpringBoot内置数据源持久化
    2022-07-07
  • 详解如何使用IntelliJ IDEA生成UML图

    详解如何使用IntelliJ IDEA生成UML图

    在软件开发中,UML(统一建模语言)是一种用于描述、构建和文档化软件系统的图形化语言,它帮助开发者以可视化的方式理解系统的结构和行为,手动绘制 UML 图可能既耗时又容易出错,所以本文给大家介绍了如何使用IntelliJ IDEA生成UML图,需要的朋友可以参考下
    2024-10-10
  • Java中如何将list转为树形结构

    Java中如何将list转为树形结构

    这篇文章主要介绍了Java中如何将list转为树形结构,本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-09-09
  • windows定时器配置执行java jar文件的方法详解

    windows定时器配置执行java jar文件的方法详解

    这篇文章主要给大家介绍了关于windows定时器配置执行java jar文件的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-11-11
  • Spring AOP中三种增强方式的示例详解

    Spring AOP中三种增强方式的示例详解

    AOP (Aspect Orient Programming),直译过来就是 面向切面编程。AOP 是一种编程思想,是面向对象编程(OOP)的一种补充。本文为大家介绍了Spring AOP中三种增强方式,感兴趣的可以了解一下
    2022-07-07
  • java依赖混乱存在的问题与解决方案

    java依赖混乱存在的问题与解决方案

    这篇文章主要为大家介绍了java依赖混乱存在的问题与解决方案,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-09-09
  • Java实现学生信息管理系统IO版本

    Java实现学生信息管理系统IO版本

    这篇文章主要为大家详细介绍了Java实现学生信息管理系统IO版本,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-04-04
  • maven如何查看jar的pom引入来源

    maven如何查看jar的pom引入来源

    这篇文章主要介绍了maven查看jar的pom引入来源,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-07-07
  • SpringBoot整合redis实现输入密码错误限制登录功能

    SpringBoot整合redis实现输入密码错误限制登录功能

    遇到这样的需求需要实现一个登录功能,并且2分钟之内只能输入5次错误密码,若输入五次之后还没有输入正确密码,系统将会将该账号锁定1小时,这篇文章主要介绍了SpringBoot整合redis并实现输入密码错误限制登录功能,需要的朋友可以参考下
    2024-02-02
  • JAVA并发图解

    JAVA并发图解

    这篇文章主要介绍了JAVA的并发,文中图解非常细致,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2021-09-09

最新评论