1. jdk动态代理原理
生成一个类A继承proxy实现要代理的接口(因为继承了proxy,所以只能代理接口了,java没法多继承),执行代理接口方法的时候,实际调用的是类A的方法。此方法内部会调用代理类的invoke方法,invoke方法会调用真实类的对应方法(反射机制),可以在invoke方法内部做一些事情。
2. cglib原理
CGLIB(Code Generation Library)是一个基于ASM的字节码生成库,它允许我们在运行时对字节码进行修改和动态生成。CGLIB通过继承方式实现代理。既然是继承就不得不考虑final的问题。我们知道final类型不能有子类,所以CGLIB不能代理final类型
3.Spring如何选择用JDK还是CGLiB?
1)当Bean实现接口时,Spring就会用JDK的动态代理。
2)当Bean没有实现接口时,Spring使用CGlib是实现。
3)可以强制使用CGlib(在spring配置中加入<aop:aspectj-autoproxy proxy-target-class="true"/>)。
4 CGLiB比JDK快?
1)使用CGLib实现动态代理,CGLib底层采用ASM字节码生成框架,使用字节码技术生成代理类,
在jdk6之前比使用Java反射效率要高。唯一需要注意的是,CGLib不能对声明为final的方法进行代理,
因为CGLib原理是动态生成被代理类的子类。
2)在jdk6、jdk7、jdk8逐步对JDK动态代理优化之后,在调用次数较少的情况下,JDK代理效率高于CGLIB代理效率,
只有当进行大量调用的时候,jdk6和jdk7比CGLIB代理效率低一点,但是到jdk8的时候,jdk代理效率高于CGLIB代理,
总之,每一次jdk版本升级,jdk代理效率都得到提升,而CGLIB代理消息确有点跟不上步伐
5 ASM?
asm是字节码增强技术,目前有如下这么多
image.png
对于需要手动操纵字节码的需求,可以使用ASM,它可以直接生产 .class字节码文件,也可以在类被加载入JVM之前动态修改类行为(如下图17所示)。ASM的应用场景有AOP(Cglib就是基于ASM)、热部署、修改其他jar包中的类等
6 运行是类动态加载
一个类已经被jvm加载之后,不允许重复加载和替换的。那如何运行期间把类给替换了呢
instrument是JVM提供的一个可以修改已加载类的类库,专门为Java语言编写的插桩服务提供支持。它需要依赖JVMTI的Attach API机制实现
7 内存溢出
CGlib 可以设置不使用代理类缓存,enhancer .setUseCache为false时,产生的代理类,都是不同的。这个时候会导致内存溢出。jdk动态代理生成的对象在堆里面,
网友评论