@annotation用在定义连接点时,对连接点进行限制。比如我们想对标注了@Deprecated的方法进行拦截,可以如下定义。

@Service
public class ServiceA {
    @Deprecated
    public void testa() {
        System.out.println("testa");
    }
}

@Service
public class ServiceB {
    public void testb() {
        System.out.println("testb");
    }
}


@Service
@Aspect
public class MyAspect {
    @Before("execution(public void com.example.demo.service.Service*.*()) && @annotation(Deprecated)")
    public void before() {
        System.out.println("拦截到了被@Deprecated标注的testa方法,没有拦截testb方法");
    }
}

运行结果如下。可以看到,按照execution中的定义,testa和testb方法都是能被拦截的,但是因为后面我们又加了@annotation的限制,所以只拦截了testa方法。

注意,@annotation只能用于拦截在方法上定义了对应注解的方法,在类上的注解@annotation是不起作用的,因为spring aop本身就是基于方法拦截的。

之所以提这一点是因为今天看代码的时候看到如下一段aop定义,我一看还以为如下定义能够拦截用@Service标注的类的所有方法。于是自己就实验了下,发现并不行。

顺便再提一个细节吧,以免有人做实验时出现我这个错误。就是我在测试的时候,如果如下定义连接点,启动的时候会报错(我用的spring-boot-starter-web进行的测试)。

分析原因是Deprecated这个注解不只我这里用到了,可能有很多类里都用到了,结果导致spring在启动时,这个切面把所有有Deprecated注解的方法都拦截了,导致可能有的类初始化有问题。所以我测试的时候加了execution(public void com.example.demo.service.Service*.*()),限制spring只拦截我测试用到的两个类。

@Before("@annotation(Deprecated)")

 

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐