Spring AOP 注解方式实现的一些“坑”
2015-11-30 本文已影响10177人
Devid
使用过Spring AOP的注解方式实现,也入过不少坑,现在做一下记录,希望有同样需求的朋友可以少走弯路
使用之前还是先过一下官方文档吧,至少可以少走弯路,不用担心英文读不懂,其实只看代码例子就能理解很多问题!
1. SpringBoot如何获得Aspect支持?
2. 我按照上面的方法开启了AOP支持,为什么始终是不走切入点方法呢?
首先仔细检查一下,Aspect类是不是少了@Component
注解(这一点很重要!),只有一个@Aspect
注解是不行的。如果还是不行,请继续往下看。
3. 拦截指定注解的规则怎么写?如拦截所有带了RequestMapping
的方法
首先声明一个写入点来匹配所有要拦截的方法名
假如拦截所有方法
@Pointcut("execution(public * *(..))")
private void aspectMethod(){}
然后配置通知方法
@Before(value = "aspectMethod() && @annotation(org.springframework.web.bind.annotation.RequestMapping)")
public String doSomething() {
}
4. 拦截指定参数名的规则该怎么写?如拦截所有方法参数中有int page
的方法
参数拦截可以用
args
或@args
,但是匹配的时候跟顺序和数量有关!
- 如果只有
page
一个参数
@Before(value = "aspectMethod() && args(page)")
public String doSomething(int page) {
}
- 如果
page
为第一个参数
@Before(value = "aspectMethod() && args(page, ..)")
public String doSomething(int page) {
}
- 如果
page
为第二个参数
@Before(value = "aspectMethod() && args(*, page, ..)")
public String doSomething(int page) {
}
说明:
- 参数的配置可以用占位符
*
和..
。
*
的意思是任意类型任意名称的一个参数
..
的意思是任意类型,任意多个参数,并且只能放到args的后面。- 通知方法的参数需要跟args中的固定参数
page
一致!- 如果要获取拦截方法的所有参数,可以用
JoinPoint
或ProceedingJoinPoint
,两者的区别:
JoinPoint
是父类,提供获取拦截方法的信息的功能,如所有参数:jp.getArgs()
ProceedingJoinPoint
是子类,只能用在@Around
中,除了提供JoinPoint
的所有功能外,还能提供方法的运行pjp.proceed()
和pjp.proceed(args)
功能。
那下面问题来了?我想匹配page参数,但是page参数的位置不定怎么办?
spring aop并不存在一个或多个的匹配,所以如果可能,尽量将配置作为第一个参数,这样写起来方便,如果真的需要匹配不确定的位置,那就多写几个匹配规则吧!
- 第一个参数:
args(page, ..)
- 第二个参数:
args(*, page, ..)
- 第三个参数:
args(*, *, page, ..)
- 以此类推
如果这样还没有满足所有需求,那么只能问问自己,编码的时候知道不知道什么叫约定大于配置
了。
5. 如果即有拦截参数,又需要注入JoinPoint
或ProceedingJoinPoint
怎么办?
将JoinPoint
或ProceedingJoinPoint
作为第一个参数,剩下的自定义的参数放到后面,如:
@Before(value = "aspectMethod() && args(page)")
public String doSomething(JoinPoint jp, int page) {
}
6. 有没有配置拦截规则示例?
有的,请查看原文:aop pointcuts examples。
这里注意,
*
用在返回值和方法参数上并不是一个或多个,而是指任意一个,如用在返回值上,表示任意类型的返回值;用在方法的参数上,指任意类型的一个参数。官方也给出了解释,在这里可能会有很多坑,大家可以注意一下!