args()的作用主要有两点:
1、切入点表达式部分如果增加了args()部分,那么目标方法除了要满足execution部分,还要满足args()对方法参数的要求,对于符合execution表达式,但不符合args参数的方法,不会被植入切面。
2、定义了args()之后,才能把目标方法的参数传入到切面方法的参数中(通过Joinpoint也可以获取参数,但当前方法是直接用切面方法参数接受)。
示例1
目标方法:
1
2
3
4
5
6
7
8
9
10
11
|
@RestController
@RequestMapping ( "/testAop" )
public class TestController {
private Logger logger = LoggerFactory.getLogger(TestController. class );
@RequestMapping ( "/helloworld" )
public String helloWorld(String id, Integer age){
System.out.println( "被代理方法正在执行" );
return null ;
}
}
|
切面方法
1
2
3
4
5
|
@After ( "execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)" )
public void after(JoinPoint point, String userId, Integer userAge){
System.out.println( "userId===========" + userId);
System.out.println( "userAge===========" + userAge);
}
|
输出结果:
被代理方法正在执行
userId===========bian1996
userAge===========24
定义了args(userId, userAge)才能把目标方法helloWorld(String id, Integer age)的参数传入到增强处理方法after的参数中,id参数对应userId,age参数对应userAge。使用的方法是按顺序一一对应,helloWorld第一个参数对args第一个参数,helloWorld第2个参数对args第2个参数。
切入点表达式部分增加了&&args(userId, userAge)部分,意味着可以在增强处理方法中定义userId、userAge两个形参------定义这两个形参时,形参类型可以随意指定,但是一旦指定,譬如这里分别是String类型和Integer类型,这两个形参类型将用于限制该切入点只匹配第一个参数类型为String,第二个参数类型为Integer的方法。
也就是,args()中的参数会和目标方法的参数除了在顺序上一一对应之外,在类型上也要对应,否则匹配失败,如下两种情况都会匹配失败。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
@RequestMapping ( "/helloworld" )
public String helloWorld(Integer id, Integer age){
System.out.println( "被代理方法正在执行" );
return null ;
}
@After ( "execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)" )
public void after(JoinPoint point, String userId, String userAge){
System.out.println( "userId===========" + userId);
System.out.println( "userAge===========" + userAge);
}
@RequestMapping ( "/helloworld" )
public String helloWorld(Integer sex, String id, Integer age){
System.out.println( "被代理方法正在执行" );
return null ;
}
@After ( "execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)" )
public void after(JoinPoint point, String userId, Integer userAge){
System.out.println( "userId===========" + userId);
System.out.println( "userAge===========" + userAge);
}
|
除此之外,使用args()表达式时还可使用如下形式:args(userId, userAge,..),这表明增强处理方法中可以通过userId, userAge来访问目标方法的参数。注意上面args表达式括号中的2点,它表示可以匹配更多参数,但是只要前两个userId, userAge参数匹配上了,目标方法就可以被匹配上。
argNames是可选的,如果没有argNames这个参数,而编译器设置了【在class文件生成变量调试信息】,则spring可以通过反射知道方法参数的名字,通过名字配对,Spring知道args(userId, userAge)表达式里面的userId和userAge,对应了增强方法public void after(JoinPoint point, String userId, Integer userAge)方法里面的userId和userAge,就是第一个示例的情况:
总结:
目标方法和args()通过参数顺序一一进行匹配
args()和增强方法通过参数名称一致进行匹配。
但是,如果设置了argNames,Spring不再使用方法参数的名字来配对,使用argNames定义的顺序来给
after(JoinPoint point, String userAge, String userId)的参数传值,例如:argNames="userId,userAge",userId在userAge前面,表示after方法第一个参数(JoinPoint 除外)是userId,第二个参数是userAge,示例如下:
目标方法
1
2
3
4
5
|
@RequestMapping ( "/helloworld" )
public String helloWorld(String id, String age){
System.out.println( "被代理方法正在执行" );
return null ;
}
|
切面方法
1
2
3
4
5
|
@After (value = "execution(* com.bxp.controller.TestController.*(..)) && args(userId, userAge)" , argNames = "userId,userAge" )
public void after(JoinPoint point, String userAge, String userId){
System.out.println( "userId===========" + userId);
System.out.println( "userAge===========" + userAge);
}
|
请求连接和输出结果
请求连接
http://localhost:8088/testAop/helloworld?age=24&id=bian1996
输出结果
被代理方法正在执行
userId===========24
userAge===========bian1996
注意:这一次两个参数的类型都给成String类型了
总结:
目标方法和args()通过参数顺序一一进行匹配
args()和argNames通过参数名称一致进行匹配
argNames和增强方法通过参数顺序一一对应。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。
原文链接:https://blog.csdn.net/b15735105314/article/details/105088734