1:简介
classLoader是一个抽象类,给定一个class的二进制文件名,ClassLoader会将会加载并且在JVM中生成构成这个类的各个数据结构,然后使其分布在JVM对应的内存区域中
类加载过程三个阶段: 加载阶段,连接阶段,初始化阶段
2:类的主动使用与被动使用
JVM虚拟机规范规定:每个类或者接口被Java程序首次主动使用时才会对其进行初始化,但是不排除JVM在运行期间提前预判并且初始化某个类
JVM规范了6种主动使用类的场景(类会被加载和初始化)
1:通过new关键字导致类的初始化,Thread t = new Thread()
2:访问类的静态变量,包括读取和更新会导致类的初始化
3:访问类的静态方法,会导致类的初始化
4:对某个类进行反射操作,会导致类的初始化
5:初始化子类会导致父类的初始化
6:启动类,执行了main函数所在的类,会导致该类的初始化
被动使用(不会被加载和初始化)
1:构造某个类的数组时并不会导致该类的初始化。
2:引用类的静态常量
1:类的加载阶段
2:类的连接阶段
2.1:验证(文件格式验证,元数据验证(语义分析),字节码验证(循环,分支),符号引用验证)
确保class文件的字节流所包含的内容符合当前JVM的规范要求。并且不会出现危害JVM自身安全的代码
2.2:准备
为静态变量设置默认值。
静态变量-->分配到方法区
实例变量-->分配到堆内存
2.3:解析
解析就是在常量池中寻找 类,接口,字段和方法的符号引用,并且将这些符号引用替换成直接引用的过程
3:类的初始化阶段
<clinit>()全称 class initialize
在初始化阶段做的最主要的一件事情就是执行 ()方法的过程,在()方法中所有的类变量都会被赋予正确的值,
也就是程序编写的时候指定的值。
eg:: private static int x = 12
此时x的值就为12了
packagejuc_ch.com.chen.test;
publicclassSingleton{
privatestaticSingletoninstance=newSingleton();//2
privatestaticintx=0;//1
privatestaticinty;
privateSingleton() {
x++;
y++;
}
publicstaticSingletongetInstance() {
returninstance;
}
publicstaticvoidmain(String[]args) {
Singletonsin=Singleton.getInstance();
System.out.println("x: "+sin.x);
System.out.println("y: "+sin.y);
}
}
//第一种情况。1在2的上面
//static是按顺序加载。.1先完成了初始化。此时x的值为0 。当2 完成初始化的时候。x++执行
//第二种情况 2在1的上面
//由于 在类的连接阶段中的准备这一时刻。此时 x 和 y的值都被赋初始值 0 。instance的初始值为null
//在初始化阶段,由于2在1的上面,按顺序执行,2先初始化 ,此时 x++,y++. 值变为 x=1,y=1
//但是,x和y后执行初始化阶段,此时x和y都被赋予正确的值(程序编写时指定的值),此时x =0; y没有指定值。那么就不赋值 y=1