• SpringAOP 切点函数详解


    一:在函数入参中使用通配符

    @AspectJ支持3种通配符

    * :匹配任意字符,但它只能匹配上下文中的一个元素.

    ..  :匹配任意字符,可以匹配上下文中多个元素,但在表示类时,必须和*联合使用,而在表示入参时则单独使用

    + :表示按类型匹配指定类的所有类,必须跟在类名后面,如com.smart.Car+ ;继承或扩展指定类的所有类,同时还包括指定类本身.

    @AspectJ函数按其是否支持通配符及支持的程度,可以分为以下3类.

    1):支持所有的通配符:execution(),within()

    2):仅支持“+”通配符:args(),this(),target()

    3):不支持通配符:@args(),within(),target();@annotation()

    此外,args(),this(),target(),@args(),@within(),@target()和@annotation()这7个函数除了可以指定类名外,也可以指定变量名,并将目标对象中的变量绑定到增强的方法中.

    二:切点函数详解

    [1]:@annotation()

    @annotation()表示标注了某个注解的所有方法.

    eg:

    package com.springboot.test;
    
    import org.aspectj.lang.annotation.AfterReturning;
    import org.aspectj.lang.annotation.Aspect;
    import org.springframework.stereotype.Component;
    
    @Aspect
    @Component
    public class TestAspect {
        @AfterReturning("@annotation(com.springboot.anno.NeedTest)")
        public void needTestFun() {
            System.out.println("needTestFun() executed!");
        }
    }

    [2]:execution()

    execution()是最常用的切点函数,其语法如下:

    execution(<修饰符模式>?<返回类型模式><方法名模式>(<参数模式>) <异常模式>?)

    除了返回类型模式,方法名模式,参数模式外,其他项都是可选的.

      (1):通过方法签名定义切点

      execution(public * *(..)):匹配所有目标类的public方法,第一个*代表返回类型,第二个*代表方法名,而..代表任意入参的方法.

      execution(* *To(..)):匹配所有以To为后缀的方法,第一个*代表返回类型,而*To代表任意以To为后缀的方法.

      (2):通过类定义切点

      execution(* com.springboot.Waiter.*(..)):匹配Waiter接口的所有方法,第一个*代表返回任意类型;com.springboot.Waiter.*(..)代表Waiter接口的所有方法,

      (3):通过类包定义切点

      在类名模式串中,“.*”表示包下所有的类,而“..*”表示包,子孙包下的所有类.

      execution(* com.smart.*(..)):匹配com.smart包下的所有类的所有方法.

      execution(* com.smart..*(..)):匹配com.smart包.子孙包下所有的类的所有方法.

      execution(* com..*.*Dao.find*(..)):匹配包名前缀为com的任何包下类名后缀为Dao的方法,方法名必须以find为前缀.如:com.smart.UserDao#findByUserId(),   com.smart.dao.ForumDao#findById()等.

      (4):通过方法入参定义切点:

      切点表达式中的方法入参部分比较复杂,可以使用“*”,“..”通配符.其中“*”表示任意类型的参数;而“..”表示任意类型的参数且参数个数不限.

      execution(* joke(String,int)):匹配joke(String str,int d)方法.

      execution(* joke(String,*):匹配目标类中的joke(),但该方法的第一个入参为String类型,第二个入参可以是任意类型.

      execution(* joke(String,..)):匹配目标类中的joke(),该方法的第一个入参为String类型,后面可以有任意个入参,且入参类型不受限制.

      execution(* joke(Object+)):匹配目标类中的joke(),方法拥有一个入参,且入参是Object类型或该类的子类.

    [3]:args()和@args()

    args():该函数接收一个类名,表示目标类方法入参对象是指定类(包含字类)时,切点匹配

      1):args(com.smart.Waiter)表示运行时入参是Waiter类型的方法, 其等价于execution(* *(com.smart.Waiter+))当然也等价于args(com.smart.Waiter+).

      2):@args() 太啰嗦,不打字了...

    [4]:within()

    通过类匹配模式串声明切点,within()函数定义的连接点是针对目标类而言的,而非针对运行期对象的类型而言,这一点和execution()是相同的.但和execution()函数不同的是,within()所指定的连接点的最小范围只能是类,二execution()所指定的连接点可以大到包,小到方法入参.所以从某种意义上说,execution()函数的功能涵盖了within()函数的功能.within()函数语法如下:

    within(<类匹配模式>)

      within(com.smart.NativeWaiter): 匹配目标类NativeWaiter的所有方法.

      within(com.smart.*):  匹配com.smart包中的所有类,但不包括子孙包,所以com.smart.service包中类的方法不匹配这个切点

      within(co.smart..*):  匹配com.smart包及子孙包中的类,所以com.smart.service,com.smart.dao,com.smart.service.forum等包中的所有类都匹配这个切点.

    [5]@within() ,@target()不解释,太啰嗦了...

    [6]target(),this()后续补充...

  • 相关阅读:
    Transformers 简介(下)
    OpenCV-Python 姿态估计 | 五十
    Transformers 简介(上)
    OpenCV-Python 相机校准 | 四十九
    在Keras中可视化LSTM
    分析师和统计学家可以和谐相处吗?
    Array类模板
    C++中 公有继承 私有继承 和保护继承的区别
    文件输入输出
    PTA(浙大数据结构,c语言)
  • 原文地址:https://www.cnblogs.com/xibushijie/p/12846042.html
Copyright © 2020-2023  润新知