分类 标签 存档 社区 博客 友链 GitHub 订阅 搜索

Spring AOP

303 浏览

ZERO

    持续更新 请关注:https://zorkelvll.cn/blogs/zorkelvll/articles/2018/11/18/1542542859691

背景

     本文主要是记录在学习 Spring AOP 过程中的一些知识点记录!

关键词:AOP 代理、AspectJ 静态代理、SpringAOP 动态代理

问题

1、什么时候使用过 JDK 动态代理和 CGLib 动态代理?

一、AOP - Aspect Orient Programming

     AOP 面向切面编程,是对面向对象编程的一种补充,常用于处理系统中各模块中的公共横切关注点,如事务管理、日志、缓存等;

    AOP 实现的关键在于 AOP 框架自动创建的 AOP 代理(静态代理、动态代理),静态代理以 AspectJ 为代表且动态代理 SpringAOP 为代表;

    AspectJ 是在编译时增强了目标对象,SpringAOP 则是在每次运行时动态地增加,区别在于二者生成 AOP 代理对象的时机不同(相对来说 AspectJ 静态代理性能更好些,但需要特定编译器处理而 SpringAOP 无需特定编译器)

    调用者类 -> 伪装类( -> 目标类),所谓 aop 就是将对目标类的调用用伪装类进行替换,即代理对象的方法 = 增强处理 + 被代理对象的方法

二、AspectJ 

     实现原理:静态代理 - 编译时增强

    所谓静态代理,就是 AOP 框架在编译阶段生成 AOP 代理类,并且 AspectJ 则是静态代理的增加(-> 所以也称为编译时增强)

三、Spring AOP   

    实现原理:动态代理 - JDK 动态代理、CGLIB 动态代理

    所谓动态代理,就是 AOP 框架不会去修改字节码,而是在内存中为方法临时生成一个 AOP 对象(包含了目标对象的全部方法)并且在特定的切点做了增强处理和回调原对象的方法(-> 所以也称为运行时增强)

    SpringAOP 的动态代理有两种方式 - JDK 动态代理和 CGLIB 动态代理

JDK 动态代理

    JDK 动态代理通过反射来接收被代理的类,并且要求被代理的类必须实现一个接口;

    JDK 动态代理的核心是 InvocationHandler 接口和 Proxy 类

CGLIB 动态代理:

    如果目标类没有实现接口,那么 SpringAOP 则会选择 CGLIB 动态代理目标类;

    CGLIB(Code Generation Library)即一个代码生成的类库,实现在运行时动态地为某个类生成其子类,也即是通过继承实现动态代理(-> 因此 final 类时无法使用 CGLIB 动态代理的)

    如果目标类没有实现接口,那么 Spring AOP 会选择使用 CGLIB 来动态代理目标类。CGLIB(Code Generation Library),是一个代码生成的类库,可以在运行时动态的生成某个类的子类,注意,CGLIB 是通过继承的方式做的动态代理,因此如果某个类被标记为 final,那么它是无法使用 CGLIB 做动态代理的。

四、核心概念   

通知 Advice:

    需要在连接点处所执行的代码,分为前置、后置、异常、最终、环绕通知;;;也即,想要的功能如安全、事务、日志等,先定义好才能仔想用的地方用一下;

连接点 JoinPoint:

    定义的可被拦截到的点(Spring 只支持方法连接点,而 AspectJ 则也支持构造器或属性字段注入);;;也即允许使用通知的地方,如方法的前、后、前后、异常;

切入点 PointCut:

    连接点的集合,由切入点表达式进行指定的需要被拦截的哪些类的哪些方法,在类运行时动态织入通知;;;也即是用切点筛选出方法(连接点)的,选中想要使用通知的那几个方法的;

切面 Aspect:

    通知 + 切入点,是对横切关注点的抽象,抽取重复的功能代码;;;也即切面是通知和切入点的结合,前者说明了做什么,后者说明在哪个地方执行,这就是一个完整的切面;

引入 introduction:

    在不修改代码的前提下,在运行期为类动态地添加一些方法或属性;;;也即把切面(通知定义的新方法属性)用到目标类中;

织入 weaving:

    把切面应用到目标对象和创建新的代理对象的过程;

目标 target:

    被织入切面的目标类,也即需要被通知的对象,只关注于其业务本身的逻辑;

五、SpringAOP 注解   

1、@Aspect :指定一个类为切面类;

2、@Pointcut(“execution(* cn.zorke.e_aop_anno..(..))”) :指定切入点表达式;

3、@Before(“pointCut_()”) :前置通知,在目标方法执行之前执行;

4、@After(“pointCut_()”) :后置通知,在目标方法执行之后执行(始终执行);

5、@AfterReturning(“pointCut_()”) :返回后通知,目标方法执行结束前执行(异常不执行);

6、@AfterThrowing(“pointCut_()”) :前置通知,在目标方法出现异常时候执行;

7、@Around(“pointCut_()”) :前置通知,环绕目标方法执行;

参考:

1、SpringAOP 概念理解(通俗易懂)

评论  
留下你的脚步
推荐阅读