浅谈Spring AOP中args()和argNames的含义

时间:2022-09-23 08:51:52

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