java反射
java反射性能问题
1. Class.forName(权限定类名)
2. 找到字节码文件
3. 通过类加载器,load字节码文件
4. 加载-连接-初始化,完成类加载
5. 加载到JVM运行时数据区(方法区)
6. 产生一个class类型的对象clazz,通过clazz.newInstance获取实例
7.
反射与new对象是100倍差距
new是JVM的指令不需经过上面的过程,JVM做了优化,性能较高
类加载过程
ClassFileStream
ClassFileParser
ClassLoad
符号引用(getName),直接引用(内存中的地址0x0101000)
类加载机制
Load -> Link(verify,prepare,resolve) -> initialize -> use ->unload
类初始化时机-initialize
获取Class 方式
Class c = User.Class; 没有初始化过程,获取类型引用
Class c = instanceObj.getClass; 对象已生成,完成初始化
Class c = UserClassLoader.loadClass("包名"); 未完成初始化
Class c = Class.forname("包名");完成初始化
Class内部元素
java.lang.class,对字节码文件内容的抽象
class对象:包名,构造函数,属性字段,方法,修饰符
JVM层使用字节码文件,OOP层面使用java.lang.class
Class c = Class.forname("包名");
c.newInstance();本质是什么?
JDK层面:通放射获取Class的constructor,invoke constructor.newInstance()
JVM层面:instanceKlass.hpp , 类型中有allocate_instance()方法,给instanceOop对象开辟空间,继承oopDesc抽象类(对应object)
内核层面?
反射会破坏单例吗?
通过放射机制可以破坏单例模式-二方包,三方库修改源码,改造
单例模式:
- 私有化构造方法
- 全局唯一的公有访问点
Spring IOC 和 DI
反射类似实现框架JUnit
读取test类的字节码,获取@test注解
Class.forname