切点函数是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. Leetcode: Convert a Number to Hexadecimal

    Given an integer, write an algorithm to convert it to hexadecimal. For negative integer, two's compl ...

  2. Error applying BeanValidation relational constraints 错误解决

    来自http://blog.csdn.net/sivyer123/article/details/9185325 在hibernate.hbm.xml中加上 <property name=&qu ...

  3. JetBrains公司介绍(Java、Python、PHP、Ruby、前端和代码测试与重构的IDE)

    JetBrains JetBrains是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄国的圣彼得堡及美国麻州波士顿都设有办公室,该公司最为人所熟知的产品是Java编程语言开发撰写时所用的集成开 ...

  4. bzoj4238 电压

    首先先直接对图进行二染色,dfs染完色后,有的边为搜索树边,有的为非树边,当非树边连接的两头的点为异色的时候,那么很明显这条非树边和树边构成的环上的边必然不可能成为答案:如果非树边的两端的点同色,那么 ...

  5. CSS_03_04_CSS伪元素选择器

    第01步:编写css代码:wei.css @charset "utf-8"; /* 伪元素选择器 :状态 效果顺序:L V H A */ a:link.lin_01{/*超链接,未 ...

  6. Java中的并发线程操作(只贴代码,内有注释)

    package com.thread1; public class LiffOff implements Runnable{ protected int countDown = 10; private ...

  7. Mysql触发器总结

    触发器(trigger):监视某种情况,并触发某种操作. 触发器创建语法四要素:1.监视地点(table) 2.监视事件(insert/update/delete) 3.触发时间(after/befo ...

  8. Deep Learning 深度学习 学习教程网站集锦

    http://blog.sciencenet.cn/blog-517721-852551.html 学习笔记:深度学习是机器学习的突破 2006-2007年,加拿大多伦多大学教授.机器学习领域的泰斗G ...

  9. css 标签 垂直居中

    <!DOCTYPE html > <html xmlns="http://www.w3.org/1999/xhtml"> <head> < ...

  10. c++ 容器(list学习总结)

    list是一个线性双向链表结构,它的数据由若干个节点构成,每一个节点都包括一个信息块(即实际存储的数据).一个前驱指针和一个后驱指针.它无需分配指定的内存大小且可以任意伸缩,这是因为它存储在非连续的内 ...