AOP 概念
AOP:Aspect Orient Programming
中文翻译:面向切面编程
作为面向对象编程的一种补充,采用非侵入式技术手段处理系统中分布于各个模块的横切关注点,比如事务管理,安全审计,日志、缓存等等。
AOP 优缺点
优点:原有代码无需更改,不影响原有业务逻辑;
缺点:通过代理方式,会有一定的性能损失;
AOP 代理
AOP 的底层实现是通过代理和反射技术,拦截正常的方法调用,在正常方法调用完成前和完成后插入切面相关的代码。具体实现有如下两种:
- 静态代理:
代表:AspectJ,直接通过修改字节码实现,编译期完成,但是需要额外的编译器支持; - 动态代理:
代表:Spring AOP,运行时动态创建代理类;
Spring AOP 代理又分为两种,JDK Proxy 和 CGLIB,分别针对接口代理和类代理。
AOP 关键点:
- Aspect:切面
对应一个类,用于实现横跨多个企业应用类的业务处理,比如事务,日志等。
通俗的话来说,其他应用类实现业务的主要逻辑(纵向),切面关注次要方面(横向插入日志,事务处理,安全审计等动作)。 - Join Point:连接点
方法执行,异常处理,改变对象的值等特定情况; - Advice:动作
与连接点和切点相关的一系列动作
对应一个方法 - Pointcut:切点
与连接点相关的表达式,匹配则执行相关动作(Advice); - Target Object:目标对象
加入相关的动作处理(Advice)的目标对象; - AOP Proxy:代理
Spring AOP 通过 JDK Proxy 或者 CGLIB 创建代理; - Weaving:织入
关联切面和目标对象的过程。可以在编译期,加载期或运行期完成;
Spring AOP 在运行期完成这个过程,AspectJ在编译期完成这个过程;
AOP 实现举例:(函数执行时间打印)
基于常用的 Spring AOP 为例,基于注解实现打印函数执行时间的切面功能。
增加依赖包
创建 Spring 项目,在 pom.xml 加入 Spring AOP 相关的类。
#pom.xml
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjrt</artifactId>
<version>1.8.14</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.14</version>
</dependency>
开启切面能力
proxyTargetClass = true,强制使用 CGLIB 代理,如果不设置针对于接口方法织入,默认使用 JDK Proxy 代理。
@Configuration
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class AspectJConfiguration {
}
定义注解
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CallTime {
}
创建切面
@Aspect
@Component
public class CallTimeAspect {
}
增加切点
@Aspect
@Component
public class CallTimeAspect {
@Pointcut("@annotation(CallTime) && execution(public org.test.service.*(..))")
public void pointCutMethod();
}
"@annotation(CallTime) && execution(public org.test.service.*(..))"
这个表达式表示只拦截加有CallTime注解的,并且是public方法,并且是 org.test.service这个包下面的类的方法,只有匹配这个表达式的方法才会被拦截,当然这个表达式可以更复杂,比如针对参数做匹配等等。
增加连接点和动作
Spring 只支持方法连接点,主要有3种方式对方法进行增强:
@Before 方法执行前;
@After 方法执行后;
@Around 方法执行过程中;
@Aspect
@Component
public class CallTimeAspect {
@Pointcut("@annotation(CallTime) && execution(public org.test.service.*(..))")
public void pointCutMethod();
@Around("pointCutMethod()")
public Object callTime(ProceedingJoinPoint joinPoint) throws Throwable {
long start = System.currentTimeMillis();
Object proceed = joinPoint.proceed();
long end = System.currentTimeMillis();
String className = proceed.getSignature().getDeclaringType().getSimpleName();
String functionName = proceed.getSignature().getName();
System.out.println(String.format("%s.%s takes %dms", className, functionName, end - start));
return proceed;
}
}
增加拦截代码
# org.test.service
public class BusinessService {
@CallTime
void doBusiness() {
...
}
}
当doBusiness在外部被调用时,就会打印执行时间了:
BusinessService.doBusiness takes XXXms
潜在的问题
- 被final修饰的类无法被改写,因此也就没法使用 AOP 进行拦截;
- 在外部调用被拦截的方法中再调用该类其他的方法,无法使用 AOP 进行拦截;
class Test {
@CallTime
public void funcA() {
this.funcB();
}
@CallTime
public void funcB() {
}
在外部调用:
new Test().funcA()
只有 funcA 会被拦截,funcB 并不会。
- 私有方法不能被外部调用,因此也就没法使用 AOP 进行拦截;