使用过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。
这里注意,
*用在返回值和方法参数上并不是一个或多个,而是指任意一个,如用在返回值上,表示任意类型的返回值;用在方法的参数上,指任意类型的一个参数。官方也给出了解释,在这里可能会有很多坑,大家可以注意一下!
文/Devid(简书作者)
原文链接:http://www.jianshu.com/p/def4c497571c
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。
原文链接:http://www.jianshu.com/p/def4c497571c
著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。