什么是 AOP
面向切面编程(aspect-oriented programming),是一种将横切关注点与业务逻辑分离的编程方式。每个横切关注点都集中在一个地方,而不是分散在多处代码中。这样使我们的服务模块更加简洁,因为它们只包含了主要关注点的代码,而次要的功能或者说辅助的功能被转移到切面中了。
AOP 主要应用场景有:
- Authentication 权限
- Caching 缓存
- Context passing 内容传递
- Error handling 错误处理
- Lazy loading 懒加载
- Debugging 调试
- logging, tracing, profiling and monitoring 记录跟踪 优化 校准
- Performance optimization 性能优化
- Persistence 持久化
- Resource pooling 资源池
- Synchronization 同步
- Transactions 事务
AOP 核心知识点
主要术语
- Aspect(切面):切入业务流程的一个独立模块,在一个应用程序可以拥有任意数量的切面。如事务管理,就是切面的一个应用例子;
- Join point(连接点):业务流程在运行过程中需要插入切面的具体位置。如执行某个特定方法或者处理异常的时候;
- Advice(通知):是切面的具体实现方法。可分为前置通知(Before)、后置通知(AfterReturning)、异常通知(AfterThrowing)、最终通知(After)和环绕通知(Around)五种。实现方法具体属于哪类通知,是在配置文件和注解中指定的;
- Pointcut(切入点):用于定义通知应该切入到哪些连接点上,不同的通知通常需要切入到不同的连接点上;
- Target(目标对象):被一个或者多个切面所通知的对象;
- Proxy(代理对象):将通知应用到目标对象之后被动态创建的对象。可以简单地理解为,代理对象为目标对象的业务逻辑功能加上被切入的切面所形成的对象;
- Weaving(切入):将切面应用到目标对象从而创建一个新的代理对象的过程。这个过程可以发生在编译期、类装载期及运行期。
通知类型
Spring AOP 主要有五种通知类型,分别是前置通知(Before)、后置通知(AfterReturning)、异常通知(AfterThrowing)、最终通知(After)和环绕通知(Around):
- Before(前置通知):在连接点之前执行的通知,但这个通知不能阻止连接点之前的执行流程(除非它抛出一个异常);
- AfterReturning(后置通知):在连接点正常执行完成之后的通知;
- AfterThrowing(异常通知):在连接点方法抛出异常时执行的通知;
- After(最终通知):在连接点执行完成之后的通知(无论是正常完成,还是抛出异常退出执行);
- Around(环绕通知):可以在连接点执行前后都执行的通知。
AOP 的两种代理方式
Spring 提供了两种方式来生成代理对象: JDKProxy 和 Cglib,具体使用哪种方式生成由 AopProxyFactory 根据 AdvisedSupport 对象的配置来决定。默认的策略是如果目标类是接口,则使用 JDK 动态代理技术,否则使用 Cglib 来生成代理。
JDK 动态接口代理
JDK 动态代理主要涉及到 java.lang.reflect 包中的两个类:Proxy 和 InvocationHandler。InvocationHandler是一个接口,通过实现该接口定义横切逻辑,并通过反射机制调用目标类的代码,动态将横切逻辑和业务逻辑编制在一起。Proxy 利用 InvocationHandler 动态创建一个符合某一接口的实例,生成目标类的代理对象。
CGLib 动态代理
CGLib 全称为 Code Generation Library,是一个强大的高性能,高质量的代码生成类库,可以在运行期扩展 Java 类与实现 Java 接口,CGLib 封装了 asm,可以再运行期动态生成新的 class。和 JDK 动态代理相比较:JDK 创建代理有一个限制,就是只能为接口创建代理实例,而对于没有通过接口定义业务方法的类,则可以通过 CGLib 创建动态代理