切点函数是AspectJ表达式语言的核心, 也是使用@AspectJ进行切面定义的难点.本小节我们通过具体的实例对切点函数进行深入学习.

1.@annotation()

  @annotation()表示标注了某个注解的所有方法,这个比较简单.

2.execution()

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

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

  除了返回类型模式, 方法名模式, 参数模式是必须的外,其他两个都不是必须的

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

      • execution(public * *(..))

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

      • execution(* *To(..))

        匹配目标类所有以To结尾的方法, 第一个*代表返回类型, *To代表所有以To结尾的方法

  (2) 通过类定义切点

      • execution(* com.bao.bao.Waiter.*(..))

        匹配Waiter接口的所有方法, 第一个*号代表返回任意类型, com.bao.bao.Waiter.*代表Waiter接口中的任意方法

      • execution(* com.bao.bao.Waiter+.*(..))

        匹配Waiter接口及其所有实现类的方法, 这个和上面的区别在于,这个不但匹配Waiter接口中定义的方法, 还匹配Waiter实现类中定义的方法(但不在Waiter接口中定义),而上面的那个只匹配Waiter接口中定义的方法

  (3) 通过类包定义切点

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

      • execution(* com.bao.bao.*(..))

        匹配com. bao.bao包下所有类的所有方法

      • execution(* com.bao.bao..*(..))

        匹配com.bao.bao包,子孙包下所有类的所有方法. ".."出现在类名中时, 后面必须跟"*"表示包,子孙包下的所有类

      • execution(* com..*.*Dao.find*(..))

        匹配包名前缀为com的任何包下类名后缀为Dao的方法, 方法名必须以find为前缀.

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

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

      • execution(* joke(String, *))

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

      • execution(* joke(String, ..)) 

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

3. args()和@args()

  args()函数的入参是类名, @args()函数的入参必须是注解类的类名.虽然args()允许在类名后面使用+通配符, 但该通配符在这里没有意义, 添加和不添加都是一样的.

  (1) args()

    该函数接收一个类名, 表示目标类的方法的入参是指定类(包括子类)时, 那么该方法匹配该切点 .如

    args(com.bao.bao.Waiter)

表示运行时入参类型是Waiter的方法, 这和execution(* *(com.bao.bao.Waiter))的区别在于后者是针对方法的签名而言, 而前者是针对方法运行时入参而言的

  (2) @args()

    该函数接受一个注解类的类名, 当方法运行时入参对象所对应的类标注了指定注解时, 该方法就匹配切点.这个切点函数的匹配规则不太容易理解...

4.within()

  within()函数定义的连接点是针对目标类而言的,而非针对运行期对象的类型而言, 这和execution()是相同的.但和execution()不同的是,execution()所指定的连接点可以大到包, 小到方法入参,而within()所指定的连接点最小范围只能是类

  (1) within(com.bao.bao.NaiveWaiter)

    匹配目标类NaiveWaiter中的所有方法, 如果切点调整为within(com.bao.bao.Waiter), 则NaiveWaiter和NaughtyWaiter中的所有方法都不匹配.

  (2) within(com.bao.bao.*)

    匹配com.bao.bao中的所有类, 但不包括子孙包

  (3) within(com.bao.bao..*)

    匹配com.bao.bao包及子孙包中的类

5.@within()和@target()

  和@annotation()及@args()函数一样, 他们也只接受注解类名作为入参,其中@target(M)匹配任意标注了@M的目标类, 而@within(M)匹配标注了@M的类及子孙类.

  @target(M)切点的匹配规则如下:

    假设NaiveWaiter标注了@Monitorable, 而其子孙类CuteNaiveWaiter没有标注@Monitorable, 则@target(com.bao.bao.Monitorable)匹配NaiveWaiter类的所有方法, 但不匹配CuteNaiveWaiter类的方法.  

  @within(M)切点的匹配规则如下: 

      假设NaiveWaiter标注了@Monitorable, 而其子孙类CuteNaiveWaiter没有标注@Monitorable, 则@within(com.bao.bao.Monitorable)匹配NaiveWaiter类的所有方法, 同时还匹配CuteNaiveWaiter类的方法.  

有一个值得注意的地方, 假如某个接口标注了@M, 则所有实现该接口的类并不匹配@within(M)

6.target()和this()

    target()切点函数通过判断目标类是否按类型匹配指定类决定连接点是否匹配,而this()则通过判断代理类是否按类型匹配指定类来决定是否和切点匹配.

  target():

    (1) target(com.bao.bao.Waiter)

      NaiveWaiter, 以及CuteNaiveWaiter的所有方法都匹配切点,包括那些未在Waiter接口中定义的方法

    

基于@AspectJ和schema的aop(三)---切点函数详解的更多相关文章

  1. (spring-第19回【AOP基础篇】)基于AspectJ和Schema的AOP

    基于AspectJ就是基于@AspectJ注解,基于Schema就是全部依靠配置文件.那么首先要了解Java注解. Java注解初探 在JDK5.0中,我们可以自定义标签,并通过Java语言的反射机制 ...

  2. 基于@AspectJ和schema的aop(一)

    在前面我们使用Pointcut和Advice描述切点和增强, 并使用Advisor整合两者描述切面.@AspectJ使用注解来描述切点和增强.两者使用的方式不同, 但是在本质上都是一样的. 我们还是用 ...

  3. 基于@AspectJ和schema的aop(二)---@AspectJ基础语法

    @AspectJ使用jdk5.0和正规的AspectJ切点表达式描述切面, 由于spring只支持方法的连接点,所以Spring只支持部分AspectJ的切点语言. 1.切点表达式函数 AspectJ ...

  4. 基于@AspectJ和schema的aop(四)---@AspectJ进阶

    @AspectJ可以使用切点函数定义切点, 我们还可以使用逻辑运算对切点进行复合运算得到复合的切点. 我们还可以对切点进行命名, 从而可以复用切点.当一个连接点匹配多个切点时, 需要考虑增强织入的顺序 ...

  5. 动态载入DLL所需要的三个函数详解(LoadLibrary,GetProcAddress,FreeLibrary)

    动态载入 DLL 动态载入方式是指在编译之前并不知道将会调用哪些 DLL 函数, 完全是在运行过程中根据需要决定应调用哪些函数. 方法是:用 LoadLibrary 函数加载动态链接库到内存,用 Ge ...

  6. 【转载】动态载入DLL所需要的三个函数详解(LoadLibrary,GetProcAddress,FreeLibrary)

    原文地址:https://www.cnblogs.com/westsoft/p/5936092.html 动态载入 DLL 动态载入方式是指在编译之前并不知道将会调用哪些 DLL 函数, 完全是在运行 ...

  7. win系统动态载入DLL所需要的三个函数详解(LoadLibrary,GetProcAddress,FreeLibrary)

    动态载入 DLL 动态载入方式是指在编译之前并不知道将会调用哪些 DLL 函数, 完全是在运行过程中根据需要决定应调用哪些函数. 方法是:用 LoadLibrary 函数加载动态链接库到内存,用 Ge ...

  8. Python第三周 函数详解

    def 函数名(): """注释说明"""" 执行逻辑体 return 返回值 定义不带参数的函数 带参数的函数 默认参数 这个是 ...

  9. Spring学习之旅(七)基于XML配置与基于AspectJ注解配置的AOP编程比较

    本篇博文用一个稍复杂点的案例来对比一下基于XML配置与基于AspectJ注解配置的AOP编程的不同. 相关引入包等Spring  AOP编程准备,请参考小编的其他博文,这里不再赘述. 案例要求: 写一 ...

随机推荐

  1. 算法训练 Torry的困惑(基本型)

    http://lx.lanqiao.org/problem.page?gpid=T129 算法训练 Torry的困惑(基本型)   时间限制:1.0s   内存限制:512.0MB      问题描述 ...

  2. poj: 2159

    简单题,看起来很凶 #include <iostream> #include <stdio.h> #include <string> #include <st ...

  3. 在TextView上加上下划线或中划线

    在做商城项目是需要用到原价格以及降价后的价格,不可避免用到下划线或中划线  textView.getPaint().setFlags(Paint. UNDERLINE_TEXT_FLAG ); //下 ...

  4. MVC模型 简介

    MVC (Modal View Controler)本来是存在于Desktop程序中的,M是指数据模型,V是指用户界面,C则是控制器.使用MVC的目的是将M和V的实现代码分离,从而使同一个程序可以使用 ...

  5. BackgroundWorker的使用方法

    http://msdn.microsoft.com/zh-cn/library/system.componentmodel.backgroundworker(VS.80).aspx Backgroun ...

  6. 使用php模拟post的几种方法

    <?phpfunction file_get_contents_post($url,$post){$options=array('http'=>array('method'=>'PO ...

  7. 【py】配置python环境

    首先,在ubuntu 10.10之后的版本中,要这样安装 sudo apt-get install python-pip python-dev build-essential sudo pip ins ...

  8. 有图有真相——关于“视频专辑:零基础学习C语言 ”

  9. 关闭 Windows 的常用端口

    netstat -ano 可以看到目前开着哪些端口 netstat -ano|findstr <端口号>   可以找到开放的端口的那条,最后还列出了 PID. 然后到任务管理器中,你可以查 ...

  10. React笔记_(3)_react语法2

    React笔记_(3)_react语法2 state和refs props就是在render渲染时,向组件内传递的变量,这个传递是单向的,只能继承下来读取. 如何进行双向传递呢? state (状态机 ...