Spring aop 切入点表达式
转自: https://blog.csdn.net/qq_36951116/article/details/79172485
切入点指示符用来指示切入点表达式目的,,在Spring AOP中目前只有执行方法这一个连接点,Spring AOP支持的AspectJ切入点指示符如下:
execution:用于匹配符合的方法;
within:用于匹配指定的类及其子类中的所有方法。
this:匹配可以向上转型为this指定的类型的代理对象中的所有方法。
target:匹配可以向上转型为target指定的类型的目标对象中的所有方法。
args:用于匹配运行时传入的参数列表的类型为指定的参数列表类型的方法;
@annotation:用于匹配持有指定注解的方法;
@within:用于匹配持有指定注解的类的所有方法;
@target:用于匹配持有指定注解的目标对象的所有方法;
@args:用于匹配运行时 传入的参数列表的类型持有 注解列表对应的注解 的方法;
AspectJ切入点支持的切入点指示符还有: call、get、set、preinitialization、staticinitialization、initialization、handler、adviceexecution、withincode、cflow、cflowbelow、if、@this、@withincode;但Spring AOP目前不支持这些指示符,使用这些指示符将抛出IllegalArgumentException异常。这些指示符Spring AOP可能会在以后进行扩展。
类型匹配语法
首先让我们来了解下AspectJ类型匹配的通配符:
*:匹配任何数量字符;
..:匹配任何数量字符的重复,如在类型模式中匹配任何数量子包;而在方法参数模式中匹配任何数量参数。
+:匹配指定类型的子类型;仅能作为后缀放在类型模式后边。
示例:
- java.lang.String 匹配String类型;
- java.*.String 匹配java包下的任何“一级子包”下的String类型;
- 如匹配java.lang.String,但不匹配java.lang.ss.String
- java..* 匹配java包及任何子包下的任何类型;
- 如匹配java.lang.String、java.lang.annotation.Annotation
- java.lang.*ing 匹配任何java.lang包下的以ing结尾的类型;
- java.lang.Number+ 匹配java.lang包下的任何Number的子类型;
- 如匹配java.lang.Integer,也匹配java.math.BigInteger
这几个示例看不懂没关系,最后还有非常多的并且更详细的示例讲解
组合切入点表达式
AspectJ使用 且(&&)、或(||)、非(!)来组合切入点表达式。
在Schema风格下,由于在XML中使用“&&”需要使用转义字符“&&”来代替之,所以很不方便,因此Spring ASP 提供了and、or、not来代替&&、||、!。
切入点使用示例
一、execution:使用“execution(方法表达式)”匹配方法执行;
模式 |
描述 |
public * *(..) |
任何公共方法的执行 |
* cn.javass..IPointcutService.*() |
cn.javass包及所有子包下IPointcutService接口中的任何无参方法 |
* cn.javass..*.*(..) |
cn.javass包及所有子包下任何类的任何方法 |
* cn.javass..IPointcutService.*(*) |
cn.javass包及所有子包下IPointcutService接口的任何只有一个参数方法 |
* (!cn.javass..IPointcutService+).*(..) |
非“cn.javass包及所有子包下IPointcutService接口及子类型”的任何方法 |
* cn.javass..IPointcutService+.*() |
cn.javass包及所有子包下IPointcutService接口及子类型的的任何无参方法 |
* cn.javass..IPointcut*.test*(java.util.Date) |
cn.javass包及所有子包下IPointcut前缀类型的的以test开头的只有一个参数类型为java.util.Date的方法,注意该匹配是根据方法签名的参数类型进行匹配的,而不是根据执行时传入的参数类型决定的,,args表示则与之相反。 如定义方法:public void test(Object obj);即使执行时传入java.util.Date,也不会匹配的; |
* cn.javass..IPointcut*.test*(..) throws IllegalArgumentException, ArrayIndexOutOfBoundsException |
cn.javass包及所有子包下IPointcut前缀类型的的任何方法,且抛出IllegalArgumentException和ArrayIndexOutOfBoundsException异常 |
* (cn.javass..IPointcutService+ && java.io.Serializable+).*(..) |
任何实现了cn.javass包及所有子包下IPointcutService接口和java.io.Serializable接口的类型的任何方法 |
@java.lang.Deprecated * *(..) |
任何持有@java.lang.Deprecated注解的方法 |
@java.lang.Deprecated @cn.javass..Secure * *(..) |
任何持有@java.lang.Deprecated和@cn.javass..Secure注解的方法 |
@(java.lang.Deprecated || cn.javass..Secure) * *(..) |
任何持有@java.lang.Deprecated或@ cn.javass..Secure注解的方法 |
(@cn.javass..Secure *) *(..) |
任何返回值类型持有@cn.javass..Secure的方法 |
* (@cn.javass..Secure *).*(..) |
任何定义方法的类型持有@cn.javass..Secure的方法 |
* *(@cn.javass..Secure (*) , @cn.javass..Secure (*)) |
任何签名带有两个参数的方法,且这个两个参数都被@ Secure标记了, 如public void test(@Secure String str1, @Secure String str1); |
* *((@ cn.javass..Secure *))或 * *(@ cn.javass..Secure *) |
任何带有一个参数的方法,且该参数类型持有@ cn.javass..Secure; 如public void test(Model model);且Model类上持有@Secure注解 |
* *( @cn.javass..Secure (@cn.javass..Secure *) , @ cn.javass..Secure (@cn.javass..Secure *)) |
任何带有两个参数的方法,且这两个参数都被@ cn.javass..Secure标记了;且这两个参数的类型上都持有@ cn.javass..Secure; |
* *( java.util.Map<cn.javass..Model, cn.javass..Model> , ..) |
任何带有一个java.util.Map参数的方法,且该参数类型是以< cn.javass..Model, cn.javass..Model >为泛型参数;注意只匹配第一个参数为java.util.Map,不包括子类型; 如public void test(HashMap<Model, Model> map, String str);将不匹配,必须使用“* *( java.util.HashMap<cn.javass..Model,cn.javass..Model> , ..)”进行匹配; 而public void test(Map map, int i);也将不匹配,因为泛型参数不匹配 |
* *(java.util.Collection<@cn.javass..Secure *>) |
任何带有一个参数(类型为java.util.Collection)的方法,且该参数类型是有一个泛型参数,该泛型参数类型上持有@cn.javass..Secure注解; 如public void test(Collection<Model> collection);Model类型上持有@cn.javass..Secure |
* *(java.util.Set<? extends HashMap>) |
任何带有一个参数的方法,且传入的参数类型是有一个泛型参数,该泛型参数类型继承与HashMap; Spring AOP目前测试不能正常工作 |
* *(java.util.List<? super HashMap>) |
任何带有一个参数的方法,且传入的参数类型是有一个泛型参数,该泛型参数类型是HashMap的基类型;如public voi test(Map map); Spring AOP目前测试不能正常工作 |
* *(*<@cn.javass..Secure *>) |
任何带有一个参数的方法,且该参数类型是有一个泛型参数,该泛型参数类型上持有@cn.javass..Secure注解; Spring AOP目前测试不能正常工作 |
二、within:使用“within(类型表达式)”用于匹配指定的类的任何方法;
注意:within只能指定类,然后该类内的所有方法都将被匹配
模式 |
描述 |
within(cn.javass..*) |
cn.javass包及子包下的任何类的任何方法执行 |
within(cn.javass..IPointcutService+) |
cn.javass包或所有子包下IPointcutService类型及子类型的任何方法 |
within(@cn.javass..Secure *) |
持有cn.javass..Secure注解的任何类型的任何方法 必须是在目标对象上声明这个注解,在接口上声明的对它不起作用 |
三、this:使用“this(type)。
this表示匹配type指定的代理对象的所有方法。this的Pointcut表达式的语法是this(type),当生成的代理对象可以转换为type指定的类型时则表示匹配,也就是说type类型只能是代理类的父类或者父接口(即目标类或者目标类的父类、父接口),
还有,基于JDK接口的代理和基于CGLIB的代理生成的代理对象是不一样的。
基于CGLIB生成的代理对象,type类型可以为目标类及目标类的父类(or父接口)..因为cglib代理所生成的代理类是目标类的子类,所以 代理对象是可以转换为目标类及目标类的父类(父接口)的。
基于jdk的代理,type为目标类的父接口时,可以匹配到该目标类的代理对象。原因:
如果type是一个类类型,那么,此时该切入点表达式无法匹配到任何代理对象,因为在jdk代理中,由于目标类与代理类是兄弟关系,代理对象是不可能转换为目标类型的,............正确的是 type只能是接口,并且 type必须是目标类所实现的接口(任意一个),这样的话,代理对象才可以转换为这个接口类型,此时才能匹配成功,也就是说,type为目标类的父接口时,可以匹配到该目标类的代理对象。
注意:(jdk代理这一段话未经测试,仅仅只是我的推导)。
综上所述,type最好就是代理类所实现的接口。。。这样不论JDK还是CGLIB都一样。
模式 |
描述 |
this(cn.javass.spring.chapter6.service.IPointcutService) |
当前AOP对象实现了 IPointcutService接口的任何方法 |
this(cn.javass.spring.chapter6.service.IIntroductionService) |
当前AOP对象实现了 IIntroductionService接口的任何方法 也可能是引入接口 |
四、target:使用“target(type)
type指的是一个类或者接口的完整包路径
功能:匹配type类型的目标对象的所有方法。即目标对象可以向上转型为type类型就算是匹配成功
模式 |
描述 |
target(cn.javass.spring.chapter6.service.IPointcutService) |
当前目标对象(非AOP代理对象)实现了 IPointcutService接口的任何方法 |
target(cn.javass.spring.chapter6.service.IIntroductionService) |
当前目标对象(非AOP代理对象) 实现了IIntroductionService 接口的任何方法 不可能是引入接口 |
五、args:使用“args(参数类型列表)”匹配当前执行的方法传入时的参数类型为指定类型的执行方法;注意是匹配传入的参数类型,不是匹配方法签名的参数类型;参数类型列表中的参数必须是类型全限定名,通配符不支持;args属于动态切入点,这种切入点开销非常大,非特殊情况最好不要使用;
args(java.util.Date)
如定义方法:public void test(Object obj);执行时传入java.util.Date,则匹配,传入的如果不是java.util.Date,则不匹配
args匹配的是传入时的的参数类型是否是指定类型
模式 |
描述 |
args (java.io.Serializable,..) |
任何一个以接受“传入参数类型为 java.io.Serializable” 开头,且其后可跟任意个任意类型的参数的方法执行,args指定的参数类型是在运行时动态匹配的 |
六、@within:使用“@within(注解类型)”匹配所以持有指定注解类型内的方法;注解类型也必须是全限定类型名;
模式 |
描述 |
@within (cn.javass.spring.chapter6.Secure) |
任何目标对象对应的类型持有Secure注解的类方法; 必须是在目标对象上声明这个注解,在接口上声明的对它不起作用 |
七、@target:使用“@target(注解类型)”匹配持有指定注解的类型的目标对象;注解类型也必须是全限定类型名;
模式 |
描述 |
@target (cn.javass.spring.chapter6.Secure) |
任何目标对象持有Secure注解的类方法; 必须是在目标对象上声明这个注解,在接口上声明的对它不起作用 |
八、@args:使用“@args(注解列表)”匹配运行时传入的参数的类型持有指定注解的方法,并且args括号内可以指定多个arg;注解类型也必须是全限定类型名;
模式 |
描述 |
@args (cn.javass.spring.chapter6.Secure) |
任何一个只接受一个参数的方法,且方法运行时传入的参数类型持有cn.javass.spring.chapter6.Secure注解 |
九、@annotation:使用“@annotation(注解类型)”匹配持有指定注解的方法;注解类型也必须是全限定类型名;
模式 |
描述 |
@annotation(cn.javass.spring.chapter6.Secure ) |
方法上持有注解 cn.javass.spring.chapter6.Secure将被匹配 |
十、bean:使用“bean(Bean id或名字通配符)”匹配特定名称的Bean对象的执行方法;Spring ASP扩展的,在AspectJ中无相应概念;
模式 |
描述 |
bean(*Service) |
匹配所有以Service命名(id或name)结尾的Bean |
十一、reference pointcut:表示引用其他命名切入点,只有@ApectJ风格支持,Schema风格不支持,如下所示:
Spring aop 切入点表达式的更多相关文章
- Java开发学习(十六)----AOP切入点表达式及五种通知类型解析
一.AOP切入点表达式 对于AOP中切入点表达式,总共有三个大的方面,分别是语法格式.通配符和书写技巧. 1.1 语法格式 首先我们先要明确两个概念: 切入点:要进行增强的方法 切入点表达式:要进行增 ...
- Java进阶知识22 Spring execution 切入点表达式
1.概述 切入点(execution ):可以对指定的方法进行拦截,从而给指定的类生成代理对象.(拦截谁,就是在谁那里切入指定的程序/方法) 格式: execution(modifiers-pat ...
- aop切入点表达式
1.切入点表达式:对指定的方法进行拦截,并且生成代理表达式. 2.拦截所有public方法 <aop:pointcut expression="execution(public * * ...
- Spring AOP切点表达式用法总结
1. 简介 面向对象编程,也称为OOP(即Object Oriented Programming)最大的优点在于能够将业务模块进行封装,从而达到功能复用的目的.通过面向对象编程,不同的模 ...
- Spring AOP execution表达式
Spring中事务控制相关配置: <bean id="txManager" class="org.springframework.jdbc.datasource.D ...
- spring AOP(切面) 表达式介绍
在 spring AOP(切面) 例子基础上对表达式进行介绍 1.添加接口删除方法 2.接口实现类 UserDaoServer 添加实现接口删除方法 3.测试类调用delUser方法 4. 输出结果截 ...
- AOP 切入点表达式
8.切入点表达式 现在我们介绍一下最重要的切入点表达式: 如上文所说,定义切入点时需要一个包含名字和任意参数的签名,还有一个切入点表达式,就是* findById*(..)这一部分. 切入点表达式的格 ...
- spring AOP excution表达式各符号意思
execution(*com.sample.service.impl..*.*(..)) 符号 含义 execution() 表达式的主题 第一个“*”符号 表示返回值的类型任意: com.sampl ...
- 使用注解匹配Spring Aop切点表达式
Spring中的类基本都会标注解,所以使用注解匹配切点可以满足绝大部分需求 主要使用@within()/@target @annotaton() @args()等... 匹配@Service类中的所有 ...
随机推荐
- MQTT 入门介绍——菜鸟教程
一.简述 MQTT(Message Queuing Telemetry Transport,消息队列遥测传输协议),是一种基于发布/订阅(publish/subscribe)模式的"轻量级& ...
- Hadoop-No.10之列簇
HBase中包含列簇(column family)的概念.列簇本质上是列的存储容器.一张表可以有一个或多个列簇.每个列簇都有自己的HFile结婚,而且在执行合并操作时,同一个表的其他列簇不受影响 在很 ...
- @SpringQueryMap注解 feign的get传参方式(转)
spring cloud项目使用feign的时候都会发现一个问题,就是get方式无法解析对象参数.其实feign是支持对象传递的,但是得是Map形式,而且不能为空,与spring在机制上不兼容,因此无 ...
- JavaMail应用--通过javamail API实现在代码中发送邮件功能
JavaMail应用 在日常开发中,可能会引用到发邮件功能,例如在持续集成中,自动化测试运行完毕,自动将测试结果以报表的形式发送邮件给相关人.那么在Java中如何实现发邮件呢? 在java EE ...
- 2018第九届蓝桥杯C/C++ A组试题答案参考
题目1 标题:分数 1/1 + 1/2 + 1/4 + 1/8 + 1/16 + .... 每项是前一项的一半,如果一共有20项,求这个和是多少,结果用分数表示出来.类似:3/2当然,这只是加了前2项 ...
- C. Planning(贪心)
C. Planning time limit per test 1 second memory limit per test 512 megabytes input standard input ou ...
- ESPCMS的CSRF添加管理员账号
ESPCMS的CSRF添加管理员账号 前言 这里开始的思路是先注入进去 然后getshell 但是不关CSRF什么事 换思路 看了一下群消息 大哥发了一个视频 后台名称admin admin12 ...
- Inter IPP 处理图像数据的方法
Inter IPP没有读取图片和保存图片的函数,需要结合opencv完成这个功能. opencv读到图片以后逐个像素点赋值给IPP显然是不可取的,方法如下: int main(int argc, ch ...
- ARTS打卡计划第五周
Algorithms: https://leetcode-cn.com/problems/longest-substring-without-repeating-characters/ 采用了map的 ...
- SSRF和XSS-filter_var(), preg_match() 和 parse_url()绕过学习
0x01:url标准的灵活性导致绕过filter_var与parse_url进行ssrf filter_var() (PHP 5 >= 5.2.0, PHP 7) filter_var — 使用 ...