Java中new和反射创建对象的区别(底层)
2020.11.17
查阅了很多资料,依然有些懵懂的地方,有错之处,还请大家多多指教。
Java中new创建对象
图来自知乎
如果我们在代码中如果写了一段
A a = new A();
在JVM中会帮你做的事情有以下:
- JVM把类的.java文件编译为一个.class的字节码文件
- 类加载器把.class文件加载进jvm的内存中,一个Class对象生成,并放入方法区中,这Class对象对于任何类都是唯一一个。
做完这些之后,才是new字段的工作:
- 判断内存中是否已经有那个唯一的Class对象
- 如果没有,则进行上述操作。
- 如果有,则在内存中申请空间开辟,即根据Class对象获取有关的构造器进行实例化,在这里我们假设是一个无参数构造,那么只需要newInstance()。
Java中使用反射创建对象
依然是上面这一幅图,但是我们这次的代码是我们最常见的反射代码
Class c = Class.forName("A的全类名");
当JVM编译到这段代码的时候,他的步骤是:
1、使用类加载,将对应.class加载入内存的方法区中,并返回Class对象。
这时候,我们可以查看这个类对象里面的构造器方法,并使用无参数构造器进行构造实例,即如下代码
Constructor constructor = c.getConstructor();
Object obj = constructor.newInstance();
用同样的图,我们可以画出来。
到这里,我们几乎可以知道无论是反射,还是New,其实都是通过类加载器对.class文件加载进内存中,创建了Class对象。‘’
那么,在其他博客中提到的动态编译和静态编译就好理解了。Java中反射属于动态编译,而new属于静态编译。
粗俗解释:
1、静态编译相当于把所有需要的东西都在初始化的时候加载了,如果程序一大,就很有可能会跑得慢。
2、动态编译,在编译的时候,需要的模块都没有编译进去,启动程序的时候,模块不会被加载而是在运行的时候,需要哪个模块就调用哪个模块。
上面的过程告诉我们,我们如果用new,那么我们要创建的类都是已经“写死”在.class文件里面了,我们无法控制JVM帮我们加载类的这一项工作。
但是如果我们用反射创建类对象,我们是相当于亲自“指导”JVM,我们“按需加载”.class文件,如果内存里面没有这个类的.class文件,那么我们用Class.forName()去叫类加载器帮忙加载就行了,而不是把程序停下来,再打一段代码,再让类加载器进行加载,从而体现出了Java的“动态性”。
参考资料有:
知乎
上一篇: new一个对象竟然不是原子操作?
推荐阅读
-
Java中new和反射创建对象的区别(底层)
-
java 中newInstance()方法和new关键字的区别
-
java 中newInstance()方法和new关键字的区别
-
&和&&的共同点和区别、Java字符含义和Java创建对象的几种方式
-
Java中new关键字和newInstance方法的区别分享
-
Java和C++通过new创建的对象有何区别?
-
透彻理解Java中Synchronized(对象锁)和Static Synchronized(类锁)的区别
-
Java中new关键字和newInstance方法的区别分享
-
透彻理解Java中Synchronized(对象锁)和Static Synchronized(类锁)的区别
-
Java中成员变量与局部变量的区别、对象类型作为方法的参数和方法的返回值