【深入浅出-JVM】(75):class 装载
过程
装载
条件
主动使用 class时
- 创建一个类的实例 (new 、反射、克隆、反序列化)
- 调用类的静态方法(invokestatic)
- 使用类或接口的静态字段(getstatic、putstatic)
- 使用 reflect反射
- 初始化子类,先初始化父类
- main方法的类
例子
被动引用不会初始化类
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-02 16:29
*/
public class Parent {
static {
System.out.println("Parent init");
}
public static int v = 100;
}
package com.mousycoder.mycode.thinking_in_jvm;
import sun.jvm.hotspot.memory.ParNewGeneration;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-02 16:29
*/
public class Child extends Parent {
static {
System.out.println("Child init");
}
}
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-02 16:32
*/
public class UseParent {
public static void main(String[] args) {
System.out.println(Child.v);
}
}
虚拟机参数 -XX:+TraceClassLoading
输出
[Loaded com.mousycoder.mycode.thinking_in_jvm.Parent from file:/Users/mousycoder/My/code/mycode/target/classes/]
[Loaded com.mousycoder.mycode.thinking_in_jvm.Child from file:/Users/mousycoder/My/code/mycode/target/classes/]
Parent init
100
[Loaded java.lang.Shutdown from /Library/Java/JavaVirtualMachines/jdk1.8.0.jdk/Contents/Home/jre/lib/rt.jar]
[Loaded java.lang.Shutdown$Lock from /Library/Java/JavaVirtualMachines/jdk1.8.0.jdk/Contents/Home/jre/lib/rt.jar]
表示 Child被加载了,但是没有被初始化
final 常量不会引起类初始化
package com.mousycoder.mycode.thinking_in_jvm;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-02 16:41
*/
public class FinalFieldClass {
public static final String constString = "CONST";
static {
System.out.println("FinalFieldClass init");
}
}
package com.mousycoder.mycode.thinking_in_jvm;
public class UseFinalField {
public static void main(String[] args) {
System.out.println(FinalFieldClass.constString);
}
}
输出
CONST
FinalFieldClass 类没有因为常量 constString 被引用而初始化,而是直接把CONST 放在常量池中
加载
例子:加载String类
package com.mousycoder.mycode.thinking_in_jvm;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* @version 1.0
* @author: mousycoder
* @date: 2019-09-03 13:38
*/
public class StringTest {
public static void main(String[] args) throws ClassNotFoundException {
Class clzStr = Class.forName("java.lang.String");
Method[] ms = clzStr.getDeclaredMethods();
for (Method m : ms) {
String mod = Modifier.toString(m.getModifiers());
System.out.println(mod + " " + m.getName() + " ( ");
Class<?>[] ps = m.getParameterTypes();
if (ps.length == 0 ){
System.out.println(')');
}
for (int i = 0; i < ps.length; i++) {
char end =i==ps.length-1 ? ')':',';
System.out.println(ps[i].getSimpleName() + end);
}
System.out.println();
}
}
}
输出
“
public equals (
Object)
public toString (
)
public hashCode (
)
public volatile compareTo (
Object)
public compareTo (
String)
public indexOf (
String,
int)`
“
验证
保证加载的字节码合法
准备
为类分配对应的内存空间,并设置初始值,此阶段不会有 java 代码执行
|类型|默认初始值|
|–|–|
|int|0|
|long|0L|
|short|(short)0|
|char|\u0000|
|boolean|false|
|reference|null|
|float|0f|
|double|0f|
解析
将类、接口、字段、方法的符号引用(字面量的引用)转成直接引用(找到方法中表的位置)
初始化
执行类的初始化方法,此方法由编译器自动生成的,由类静态成员的赋值语句以及static 语句块合并产生
其中函数是带锁线程安全的,可能会导致死锁。