在前面的文章:【每天学点Spring】Spring @Aspect学习 讲了Spring通过配置@Aspect来定义切面相关。本文讲的是更底层的Spring AOP APIs,对于一般的应用程序来说,使用@Aspect足以。
我主要是看到Transaction相关的源码,发现TransactionInterceptor
实现了MethodInterceptor
,于是整理下这块内容。
官网文档:https://docs.spring.io/spring-framework/docs/current/reference/html/core.html#aop-api
1. 通过实现接口MethodInterceptor来定义我们的AOP类
1.1 MethodInterceptor接口
在spring-aop jar包中,有个类叫org.aopalliance.intercept.MethodInterceptor
,它主要有一个方法叫invoke(invocation)
:
1.2 实现自己的MethodInterceptor
上述的invoke(invocation)
方法相当于@Aspect
中的Around
,即我们可以自定义一个类并实现它,和@Around
描述的方法相似,我们可以在invocation.proeed()
方法的前面做一些事,相当于@Before
,也可以在proceed()后面做一些事,相当于@After
或@AfterReturning
,如果用try-catch,那么在catch中可以做一些@AfterThrowing
的逻辑。
@Component
public class DebugInterceptor implements MethodInterceptor {
public Object invoke(MethodInvocation invocation) throws Throwable {
log.info("Before: invocation = {}", invocation);
Object rval = invocation.proceed();
log.info("Invocation returned");
return rval;
}
}
1.3 定义切点
在org.springframework.aop
下有接口叫:Pointcut
,如果我们想要定义自己的切点规则(注意,不是切点),那么可以实现这个接口。
public interface Pointcut {
ClassFilter getClassFilter();
MethodMatcher getMethodMatcher();
//Canonical Pointcut instance that always matches.
Pointcut TRUE = TruePointcut.INSTANCE;
}
我们可以使用AspectJ的规则类:AspectJExpressionPointcut
,我们可以set expression到这个规则类中。然后将#1.2中的类似@Around功能的DebugInterceptor
类与本章节的AspectJExpressionPointcut
联系起来。
创建MyProxyFactoryBean配置类,并创建defaultPointcutAdvisor::
@Configuration
public class MyProxyFactoryBean {
@Bean
public DefaultPointcutAdvisor defaultPointcutAdvisor(DebugInterceptor debugInterceptor) {
AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();
pointcut.setExpression("@annotation(com.aspect.sample.aop.UserUpdate)");
return new DefaultPointcutAdvisor(pointcut, debugInterceptor);
}
}
上述的bean,如果用@Aspect
中的@Around
切面来表示,即我们想要实现功能:
1.4 测试
我们试图通过UserController
--> UserService.update()
,这个方法被标注了我们自定义的annotation:@UserUpdate
,正因为此,才会被我们上述的debugInterceptor
给捕捉到。
2. TransactionInterceptor是如何创建的
Spring Transaction的其中一个核心类TransactionInterceptor
,那么围绕它,是怎么setup起来的,以及核心的方法是哪个?
首先可以看到TransactionInterceptor
类的定义,和我们上述的DebugInterceptor
一样,实现了MethodInterceptor
接口:
那么每次切面被调用的方法和#1.2一样:invoke(MethodInvocation invocation)
。
【到此为止,已经介绍完了所有跟MethodInterceptor
相关的应用了。即我们的声明式事务@Transactional
的核心类:TransactionInterceptor
实现了MethodInterceptor,其中最最重要的方法即为:invoke(invocation)】
网友评论