本文主要是介绍Spring5学习笔记(九)AOP编程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
AOP概念
AOP:面向切面编程。
以切面为基本单位的程序开发,通过切面间的彼此协同,相互调用,完成程序的构建。
切面=切入点+额外功能(动态代理切入点和额外功能的组装就是切面)
OOP:面向对象编程。
以对象为基本单位的程序开发,通过对象间的彼此协同,相互调用,完成程序的构建
POP:面向过程编程。
过程就是方法或函数。以过程为基本单位的程序开发,通过过程间的彼此协同,相互调用,完成程序的构建。
AOP的概念:
本质就是Spring 的动态代理开发,通过代理类为原始类增加额外功能。
好处:利于原始类的维护
AOP不可能取代OOP,是OOP的补充。
AOP编程的开发步骤
- 原始对象
- 额外功能
- 切入点
- 组装切面
和动态代理开发是一致的
切面的名词解释
多个切入点+相同的性质(额外功能)组成一个面。
AOP底层实现原理
核心问题
-
AOP如何创建动态代理类(动态字节码技术)
proxy.newproxyInstance()函数创建代理。
原始对象自己创建的。
参数Invocationhandler用来处理额外功能,运行原始方法,返回原始对象的返回值。
interfaces用来控制代理对象和原始对象实现相同的接口
classloader的作用:
通过类加载器把对应的类的字节码文件加载JVM。
通过类加载器创建类的class对象,进而创建这个类的对象。
如何获得类加载器:虚拟机为每一个类的class文件自动分配与之对应的classloader。
因为动态代理类没有class文件,所以JVM不会为其分配类加载器,
所以要借用一个类加载器,就是传入的那个参数,借谁的都行。
动态代理类不经过加载过程,直接把代理类字节码直接写进JVM。
通过借来的加载器创建class对象,进而创建代理对象。
最终的实现。
-
spring工厂如何实现通过原始对象的ID获取代理对象
Cglib动态代理
JDK动态代理要求实现相同的接口:1.方法一致,迷惑调用者。2.代理类提供新的实现,额外功能+原始方法
如果原始类没有实现任何的接口怎么办呢?继承原始类。保证了方法一致,加入额外功能用super调用原始类。
同样需要类加载器。
接口不需要了,因为指定父类也就是原始对象,
额外功能还是有。
动态代理总结
JDK 接口创建代理的实现类
Cglib动态代理 Enhancer 通过集成父类创建代理类
Spring工厂如何加工原始对象
spring通过原始对象ID值获取代理对象,依靠BeanPostProcessor,在after中创建代理对象返回。
编码:(没有引入切入点模拟原理)
基于注解的AOP编程
- 原始对象
- 额外功能
- 切入点
- 组装切面
核心就是切面,切面也可以看做一个类。
@Aspect注解
代表类是切面类。
需要切入点+额外功能组装切面。
@Around注解
加入aroud等价于原来的invoke,额外方法。
切入点是注解中的参数,就是切入点的新的表达方式。
额外方法就是加around注解的方法,参数有变化,仍然需要调用原始方法,返回原始方法的返回值。
配置文件只需要写切面即可。
注解AOP的细节
- 切入点复用:把切入点表达式的内容提取到函数中,统一管理,便于维护,简化编程。
@Pointcut注解
- 动态代理的创建方式
默认AOP编程是jdk的实现方式。
基于注解的: 可以通过配置修改。true就是cjlib,false就是jdk。
传统开发:配置修改。位置不同,内容相同。
AOP开发中的一个坑
类切入,类中函数可能互相调用。只有最外层方法加入了额外功能,内部方法调用的都是原始方法。
这里通过this来调用login,并不会触发额外功能。因为this是原始对象,不是代理对象。
可以实现一个接口,获知工厂(不要重新创建一个工厂,因为是重量级对象)
然后通过工厂获取代理类,调用代理类的login方法,不要 调用原始类的方法。
AOP阶段总结
这篇关于Spring5学习笔记(九)AOP编程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!