类的加载、连接与初始化
在Java代码中,类型的加载、连接与初始化过程都是在程序运行期间完成的
-
加载:查找并加载类的二进制数据
-
连接
- -验证:确保被加载的类的正确性
- -准备:为类的静态变量分配内存,并将其初始化为默认值
- -解析:把类中的符号引用转换为直接引用
-
初始化:为类的静态变量赋予正确的初始化
例如以下代码:
/*
在连接阶段,将a分配内存,并将其初始化为0,在初始化阶段将a赋值为1。
*/
class Test {
Public static int a = 1;
}
-
java程序对类的使用方式可分为两种
- -主动使用
- -被动使用
-
所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才初始化他们
-
主动使用(七种)
- -创建类的实例
- -访问某个类或接口的静态变量,或者对该静态变量赋值
- -调用类的静态方法
- -反射(如Class.forName(“com.test.Test”))
- -初始化一个类的子类
- -Java虚拟机启动时被标明为启动类的类(Java Test)
- JDK1.7开始提供的动态语言支持:
java.lang.invoke.MethodHandle实例的解析结果REF_getStatic, REF_putStatic,REF_invokeStatic句柄对应的类没有初始化则初始化
-
除以上七种情况,其他使用Java类的方式都被看作是对类的被动使用,都不会导致类的初始化
-
类的加载指的是将类的.class文件中的二进制数据读取到内存中,将其放在运行时数据区的方法内,然后在内存中创建一个java.lang.Class 对象(规范并未说明Class对象位于那里,HotSpot虚拟机将其放在方法区中)用来封装类在方法区内的数据结构
-
加载.class 文件的方式
- -从本地文件系统中直接加载
- -通过网络下载.class文件
- -从Zip,jar等归档文件中加载.class文件
- -从专有数据库中提取.class文件
- -将Java源文件动态编译为.class文件
-
以下为Java主动使用举例:
package com.finedo.jvm.classloader;
/*
对于静态字段来说,只有直接定义了该字段的类才会被初始化;
当一个类在初始化时,要求其父类全部都已经初始化完毕了
*/
public class Mytest1 {
public static void main(String[] args) {
// System.out.println(MyChild1.str); // 运行结果 MyParent1 static block hello world
System.out.println(MyChild1.str2); //运行结果 MyParent1 static block MyChild1 static block welcome
}
}
class MyParent1 {
public static String str = "hello world";
static {
System.out.println("MyParent1 static block");
}
}
class MyChild1 extends MyParent1 {
public static String str2 = "welcome";
static {
System.out.println("MyChild1 static block");
}
}
-
在idea中加入以下 -XX:+TraceClassLoading(用于追踪类加载信息并打印出来):
-
输入结果显示我们定义的类(MyParent1,MyChild1)已被jvm加载: