Java 对象的创建以及类加载

1.  对象的创建的过程:

   类加载检查—>分配内存—>初始化零值—>设置对象头—>执行 init 。

   1.类加载检查: 虚拟机遇到一条 new 指令时,首先将去检查这个指令的参数是否能在常量池中定位到这个类的符号引用,并且检查这个符号引用代表的类是否已被加载过、解析和初始化过。如果没有,那必须先执行相应的类加载过程。

   2.分配内存:类加载检查通过后,接下来虚拟机将为新生对象分配内存。分配方式有:指针碰撞 和 空闲列表两种。在实际开发过程中,创建对象是很频繁的事情,作为虚拟机来说,必须要保证线程是安全的。

   3.初始化零值:虚拟机将分配到的内存空间都初始化为零值。

   4.设置对象头:虚拟机要对对象进行必要的设置,例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希吗、对象的 GC 分代年龄等信息。 这些信息存放在对象头中。

   5.执行 init :在上面工作都完成之后,从虚拟机的视角来看,一个新的对象已经产生了,但从 Java 程序的视角来看,对象创建才刚开始,所以要执行程序写的 init 方法。

2.  类加载的过程:

   Java类加载器基于三个机制:委托、可见性和单一性。委托就是将加载一个类的请求交给父类加载器,如果没有父类加载器或者父类加载器不给它加载再加载它,可见性是父类加载器不能看到子类的加载的类,但是子类的加载能看到父类的加载的类。单一性是指仅加载一个类一次,这是由委托机制确保子类加载器不会再次加载父类加载器加载过的类。

   为什么要双亲委派?保证java类库中的类不受用户类影响,防止用户自定义一个类库中的同名类,引起问题。

   我们常见的有三种加载器:Bootstrap类加载器(加载java核心类库)、Extension类加载器(加载 Java 的扩展库)和System类加载器(Java 应用的类路径来加载 Java 类)

   注意:使用不同的类加载器,你可以从不同的源地址加载同一个类,它们被视为不同的类。

   类加载主要分为以下五个步骤:加载,验证,准备,解析,初始化。   

   加载:程序运行之前jvm会把编译完成的.class二进制文件加载到内存,供程序使用,用到的就是类加载器classLoader。

   验证:保证类加载的准确性,看是否是javac 编译的class 文件。

   准备:为类的静态变量分配内存,将其初始化为默认值。

   解析:把类中的符号引用转化为直接引用。

   初始化:为类的静态变量赋予正确的初始值,上述的准备阶段为静态变量赋予的是虚拟机默认的初始值,此处赋予的才是程序编写者为变量分配的真正的初始值。

3.  自己写个String能加载吗

   答案:加载过程中会先检查类是否被已加载,检查顺序是自底向上,从Custom ClassLoader到BootStrap ClassLoader逐层检查,只要某个classloader已加载就视为已加载此类,保证此类只所有ClassLoader加载一次。而加载的顺序是自顶向下,也就是说当发现这个类没有的时候会先去让自己的父类去加载,父类没有再让儿子去加载,那么在这个例子中我们自己写的String应该是被Bootstrap ClassLoader加载了,所以App ClassLoader就不会再去加载我们写的String类了,导致我们写的String类是没有被加载的。但在应用的时候,需要用自己的类加载器去加载,否则,系统的类加载器永远只是去加载 jre.jar 包中的那个 java.lang.String。双亲委派机制是可以打破的, 所以,这些类都是可以重写的。 但是JVM出于安全的考虑, 你自己写的类 , 不能用java.* 开头, 否则都不会被加载。 懂了吧, 总结一下就是: 可以写, 但是包名要改。