欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

Java中new和反射创建对象的区别(底层)

程序员文章站 2024-03-12 15:07:02
...

2020.11.17
查阅了很多资料,依然有些懵懂的地方,有错之处,还请大家多多指教。

Java中new创建对象

Java中new和反射创建对象的区别(底层)图来自知乎

如果我们在代码中如果写了一段

A a = new A();

在JVM中会帮你做的事情有以下:

  1. JVM把类的.java文件编译为一个.class的字节码文件
  2. 类加载器把.class文件加载进jvm的内存中,一个Class对象生成,并放入方法区中,这Class对象对于任何类都是唯一一个。

做完这些之后,才是new字段的工作:

  1. 判断内存中是否已经有那个唯一的Class对象
  2. 如果没有,则进行上述操作。
  3. 如果有,则在内存中申请空间开辟,即根据Class对象获取有关的构造器进行实例化,在这里我们假设是一个无参数构造,那么只需要newInstance()。

Java中使用反射创建对象

依然是上面这一幅图,但是我们这次的代码是我们最常见的反射代码

Class c = Class.forName("A的全类名");

当JVM编译到这段代码的时候,他的步骤是:
1、使用类加载,将对应.class加载入内存的方法区中,并返回Class对象。
这时候,我们可以查看这个类对象里面的构造器方法,并使用无参数构造器进行构造实例,即如下代码

Constructor constructor = c.getConstructor();
Object obj = constructor.newInstance();

用同样的图,我们可以画出来。
Java中new和反射创建对象的区别(底层)到这里,我们几乎可以知道无论是反射,还是New,其实都是通过类加载器对.class文件加载进内存中,创建了Class对象。‘’

那么,在其他博客中提到的动态编译和静态编译就好理解了。
Java中反射属于动态编译,而new属于静态编译。

粗俗解释:

1、静态编译相当于把所有需要的东西都在初始化的时候加载了,如果程序一大,就很有可能会跑得慢。

2、动态编译,在编译的时候,需要的模块都没有编译进去,启动程序的时候,模块不会被加载而是在运行的时候,需要哪个模块就调用哪个模块。

上面的过程告诉我们,我们如果用new,那么我们要创建的类都是已经“写死”在.class文件里面了,我们无法控制JVM帮我们加载类的这一项工作。

但是如果我们用反射创建类对象,我们是相当于亲自“指导”JVM,我们“按需加载”.class文件,如果内存里面没有这个类的.class文件,那么我们用Class.forName()去叫类加载器帮忙加载就行了,而不是把程序停下来,再打一段代码,再让类加载器进行加载,从而体现出了Java的“动态性”。

参考资料有:
知乎