类初始化流程

一、类的初始化顺序
(静态变量、静态初始化块:决于它们在类中出现的先后顺序)>(变量、初始化块:决于它们在类中出现的先后顺序)>构造器
原理:
1、加载类信息。在实例化对象之前,类的装载器会找到需要加载的类class文件,进行类的加载(有父类的会先加载父类),一旦加载到最根上的基类,就会对基类的静态变量和静态初始化块进行初始化;
2、当所有类信息加载完毕就会执行main()主方法,然后执行new class(),对类进行实例化,首先对变量和、初始化块以及类的构造函数进行初始化(有父类的首先会对父类进行初始化,多个父类递归的方式)
有父类的加载顺序:
父类--静态变量
父类--静态初始化块
子类--静态变量
子类--静态初始化块
子类main方法
父类--变量
父类--初始化块
父类--构造器
子类--变量
子类--初始化块
子类--构造器

二、类加载器双亲委派模型(ClassLoader抽象类)
1、双亲委派模型,约定类加载器的加载机制


BootstrapClassLoader,由C++编写嵌套在JVM内部,负责加载“JAVA_HOME/lib”目录中的所有类型,或者由“-Xbootclasspath”指定路径中的所有类型。
ExtClassLoader和AppClassLoader都继承至ClassLoader抽象类,由Java编写;ExtClassLoader负责加载“JAVA_HOME/lib/ext”目录下的所有类型。
AppClassLoader负责加载ClassPath目录中的所有类型。
2、双亲委派模型的工作过程
双亲委派模型的工作过程是:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当父加载器反馈自己无法完成这个加载请求(它的搜索范围中没有找到所需的类)时,子加载器才会尝试自己去加载。
3、使用这种方式的好处
能够有效确保一个类的全局唯一性,当程序中出现多个全限定名相同的类时,类加载器在执行加载时,始终只会加载其中的某一个类。
4、实现双亲委派的代码java.lang.ClassLoader的loadClass()方法之中
逻辑清晰易懂:先检查是否已经被加载过,若没有加载则调用父加载器的loadClass()方法,若父加载器为空则默认使用启动类加载器作为父加载器。如果父类加载失败,抛出ClassNotFoundException异常后,再调用自己的findClass()方法进行加载。
5、双亲委派机制只是Java虚拟机规范建议采用的加载机制,在tomcat中扥实现有所不同
实际在tomcat中,类加载器所采用的加载机制与传统的双亲委派模型有一定的区别,当缺省的类加载器接收到一个类的加载任务时,首先会去由它自行加载,当它加载失败时,才会将类的加载任务委派给它的超类加载器去执行。
6、自定义类加载器
程序中如果没有显式指定类加载器的话,默认是AppClassLoader来加载,它负责加载ClassPath目录中的所有类型,如果被加载的类型并没有在ClassPath目录中时,抛出java.lang.ClassNotFoundException异常。
JDK1.2之后已不提倡用户再去覆盖loadClass()方法,而应当把自己的类加载逻辑写到findClass()方法中,在loadClass()方法的逻辑里如果父类加载失败,则会调用自己的findClass()方法来完成加载,这样就可以保证新写出来的类加载器是符合双亲委派规则的。

三、不进行类信息的初始化情况
1、通过类引用静态常量
在编译的时候,常量(static final 修饰的)会存入调用类的常量池【这里说的是main函数所在的类的常量池】,调用的时候本质上没有引用到定义常量的类,而是直接访问了自己的常量池。
2、类引用静态变量的情况
a、子类引用父类的静态变量会导致父类的类信息初始化,但是子类的类信息不进行初始化;
b、子类引用自己的静态变量,会导致自己和父类的类信息都进行初始化
3、通过数组定义来引用类,不会触发此类的初始化
SubClass[] sca = new SubClass[10];

四、类初始化的6种时机
(1)为一个类型创建一个新的对象实例时(比如new、反射、序列化)
(2)调用一个类型的静态方法时(即在字节码中执行invokestatic指令)
(3)调用一个类型或接口的静态字段,或者对这些静态字段执行赋值操作时(即在字节码中,执行getstatic或者putstatic指令),不过用final修饰的静态字段除外,它被初始化为一个编译时常量表达式
(4)调用JavaAPI中的反射方法时(比如调用java.lang.Class中的方法,或者java.lang.reflect包中其他类的方法)
(5)初始化一个类的派生类时(Java虚拟机规范明确要求初始化一个类时,它的超类必须提前完成初始化操作,接口例外)
(6)JVM启动包含main方法的启动类时。
数组本身并不是由类加载器负责创建,而是由JVM在运行时根据需要而直接创建的,但数组的元素类型仍然需要依靠类加载器去创建。

五、类加载过程:加载、连接、初始化

六、Java对象分配简要流程

参考:https://segmentfault.com/a/1190000004689988

  • 2
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值