一文教你读懂JVM类加载机制
- 加载
- 连接(验证,准备,解析)
- 初始化
I.类加载流程
1. 加载
- 根据类的全局限定名找到.class文件,生成对应的二进制字节流。
- 将静态存储结构转换为运行时数据结构,保存运行时数据结构到JVM内存方法区中。
- JVM创建java.lang.Class类型的对象,保存于堆(Heap)中。利用该对象,可以获取保存于方法区中的类信息,例如:类名称,父类名称,方法和变量等信息。
package com.demo; import java.lang.reflect.Field;
import java.lang.reflect.Method; public class ClassLoaderExample {
public static void main(String[] args) {
StringOp stringOp = new StringOp(); System.out.println("Class Name: " + stringOp.getClass().getName());
for(Method method: stringOp.getClass().getMethods()) {
System.out.println("Method Name: " + method.getName());
}
for (Field field: stringOp.getClass().getDeclaredFields()) {
System.out.println("Field Name: " + field.getName());
}
}
}
package com.demo; public class StringOp {
private String displayName;
private String address; public String getDisplayName() {
return displayName;
} public String getAddress() {
return address;
}
}
output:
Class Name: com.demo.StringOp
Method Name: getAddress
Method Name: getDisplayName
Field Name: displayName
Field Name: address
StringOp stringOp1 = new StringOp();
StringOp stringOp2 = new StringOp();
System.out.println(stringOp1.getClass() == stringOp2.getClass());
//output: true
2. 连接
2.1 验证
- 文件格式:验证文件的格式是否符合规范,如果符合规范,则将对应的二进制字节流存储到JVM内存的方法区中;否则抛出java.lang.VerifyError异常。
- 元数据:对字节码的描述信息进行语义分析,确保符合Java语言规范。例如:是否有父类;是否继承了不允许继承的类(final修饰的类);如果是实体类实现接口,是否实现了所有的方法;等。。
- 字节码:验证程序语义是否合法,确保目标类的方法在被调用时不会影响JVM的正常运行。例如int类型的变量是否被当成String类型的变量等。
- 符号引用:目标类涉及到其他类的的引用时,根据引用类的全局限定名(例如:import com.demo.StringOp)能否找到对应的类;被引用类的字段和方法是否可被目标类访问(public, protected, package-private, private)。这里主要是确保后续目标类的解析步骤可以顺利完成。
2.2 准备
public class CustomClassLoader {
//加载CustomClassLoader类时,便会为var1变量分配内存
//准备阶段,var1赋值256
public static final int var1 = 256;
//加载CustomClassLoader类时,便会为var2变量分配内存
//准备阶段,var2赋值0, 初始化阶段赋值128
public static int var2 = 128;
//实例化一个CustomClassLoader对象时,便会为var1变量分配内存和赋值
public int var3 = 64;
}
数据类型
|
默认值
|
int
|
0
|
float
|
0.0f
|
long
|
0L
|
double
|
0.0d
|
short
|
(short)0
|
char
|
'\u0000'
|
byte
|
(byte)0
|
String
|
null
|
boolean
|
false
|
ArrayList
|
null
|
HashMap
|
null
|
2.3 解析
- 符号引用(Symbolic Reference):描述所引用目标的一组符号,使用该符号可以唯一标识到目标即可。比如引用一个类:com.demo.CustomClassLoader,这段字符串就是一个符号引用,并且引用的对象不一定事先加载到内存中。
- 直接引用(Direct Reference):直接指向目标的指针,相对偏移量或者一个能间接定位到目标的句柄。根据直接引用的定义,被引用的目标一定事先加载到了内存中。
3. 初始化
- <init>():对象构造器方法,用于初始化实例对象
- 实例对象的constructor(s)方法,和非静态变量的初始化;
- 执行new创建实例对象时使用。
- <clinit>():类构造器方法,用于初始化类
- 类的静态语句块和静态变量的初始化;
- 类加载的初始化阶段执行。
public class ClassLoaderExample {
private static final Logger logger = LoggerFactory.getLogger(ClassLoaderExample.class);//<clinit>
private String property = "custom"; //<init> //<clinit>
static {
System.out.println("Static Initializing...");
} //<init>
ClassLoaderExample() {
System.out.println("Instance Initializing...");
} //<init>
ClassLoaderExample(String property) {
this.property = property;
System.out.println("Instance Initializing...");
}
}
Code:
0 aload_0 //将局部变量表中第一个引用加载到操作树栈
1 invokespecial #1 <java/lang/Object.<init>> //调用java.lang.Object的实例初始化方法
4 aload_0 //将局部变量表中第一个引用加载到操作树栈
5 ldc #2 <custom> //将常量custom从常量池推送至栈顶
7 putfield #3 <com/kaiwu/ClassLoaderExample.property> //设置com.kaiwu.ClassLoaderExample实例对象的property字段值为custom
10 getstatic #4 <java/lang/System.out> //从java.lang.System类中获取静态字段out
13 ldc #5 <Instance Initializing...> //将常量Instance Initializing从常量池第5个位置推送至栈顶
15 invokevirtual #6 <java/io/PrintStream.println> //调用java.io.PrintStream对象的println实例方法
18 return //返回
Code:
0 aload_0 //将局部变量表中第一个引用加载到操作树栈
1 invokespecial #1 <java/lang/Object.<init>> //调用java.lang.Object的实例初始化方法
4 aload_0 //将局部变量表中第一个引用加载到操作树栈
5 ldc #2 <custom> //将常量custom从常量池第二个位置推送至栈顶
7 putfield #3 <com/kaiwu/ClassLoaderExample.property> //将常量custom赋值给com.kaiwu.ClassLoaderExample实例对象的property字段
10 aload_0 //将局部变量表中第一个引用加载到操作树栈
11 aload_1 //将局部变量表中第二个引用加载到操作树栈
12 putfield #3 <com/kaiwu/ClassLoaderExample.property> //将入参property赋值给com.kaiwu.ClassLoaderExample实例对象的property字段
15 getstatic #4 <java/lang/System.out> //从java.lang.System类中获取静态字段out
18 ldc #5 <Instance Initializing...> //将常量Instance Initializing从常量池第5个位置推送至栈顶
20 invokevirtual #6 <java/io/PrintStream.println> //调用java.io.PrintStream对象的println实例方法
23 return //返回
Code:
0 ldc #7 <com/kaiwu/ClassLoaderExample> //将com.kaiwu.ClassLoaderEexample的class_info常量从常量池第七个位置推送至栈顶
2 invokestatic #8 <org/slf4j/LoggerFactory.getLogger> //从org.slf4j.LoggerFactory类中获取静态字段getLogger
5 putstatic #9 <com/kaiwu/ClassLoaderExample.logger> //设置com.kaiwu.ClassLoaderExample类的静态字段logger
8 getstatic #4 <java/lang/System.out> //从java.lang.System类中获取静态字段out
11 ldc #10 <Static Initializing...> //将常量Static Initializing从常量池第10个位置推送至栈顶
13 invokevirtual #6 <java/io/PrintStream.println> //调用java.io.PrintStream对象的println实例方法
16 return //返回
II. 类加载器
1. 类加载器ClassLoader
public static void printClassLoader() {
// StringOP:自定义类
System.out.println("ClassLoader of StringOp: " + StringOp.class.getClassLoader());
// com.sun.javafx.binding.Logging:Java核心类扩展的类
System.out.println("ClassLoader of Logging: " + Logging.class.getClassLoader());
// java.lang.String: Java核心类
System.out.println("ClassLoader of String: " + String.class.getClassLoader());
}
output:
ClassLoader of StringOp: sun.misc.Launcher$AppClassLoader@18b4aac2
ClassLoader of Logging: sun.misc.Launcher$ExtClassLoader@7c3df479
ClassLoader of String: null
- 启动类加载器:本地代码(C++语言)实现的类加载器,负责加载JDK内部类(通常是$JAVA_HOME/jre/lib/rt.jar和$JAVA_HOME/jre/lib目录中的其他核心类库)或者-Xbootclasspath选项指定的jar包到内存中。该加载器是JVM核心的一部分,以本机代码编写,开发者无法获得启动类加载器的引用,所以上述java.lang.String类的加载为null。此外,该类充当所有其他java.lang.Class Loader实例共同的父级(区别为是否为直接父级),它加载所有直接子级的java.lang.ClassLoader类(其他子类逐层由直接父级类加载器加载)。
- 扩展类加载器:启动类加载器的子级,由Java语言实现的,用来加载JDK扩展目录下核心类的扩展类(通常是$JAVA_HOME/lib/ext/*.jar)或者-Djava.ext.dir系统属性中指定的任何其他目录中存在的类到内存中。由sun.misc.Launcher$ExtClassLoader类实现,开发者可以直接使用扩展类加载器。
- 应用/系统类加载器:扩展类加载器的子级,负责将java -classpath/-cp($CLASSPATH)或者-Djava.class.path变量指定目录下类库加载到JVM内存中。由sun.misc.Launcher$AppClassLoader类实现,开发者可以直接使用系统类加载器。
2. 类加载器的类图关系





3. 双亲委派机制
- 使用双亲委派模式可以避免类的重复加载:当父级加载器已经加载了目标类,则子加载器没有必要再加载一次。
- 避免潜在的安全风险:启动类加载器是所有其他加载器的共同父级,所以java的核心类库不会被重复加载,意味着核心类库不会被随意篡改。例如我们自定义名为java.lang.String的类,通过双亲委派模式进行加载类,通过上述流程图,启动类加载器会发现目标类已经加载,直接返回核心类java.lang.String,而不会通过应用/系统类加载器加载自定义类java.lang.String。当然,一般而言我们是不可以加载全局限定名与核心类同名的自定义类,否则会抛出异常:java.lang.SecurityException: Prohibited package name: java.lang。

- 当加载器收到加载类的请求时,首先会根据该类的全局限定名查目标类是否已经被加载,如果加载则万事大吉;
- 如果没有加载,查看是否有父级加载器,如果有则将加载类的请求委托给父级加载器;
- 依次递归;
- 直到启动类加载器,如果在已加载的类中依旧找不到该类,则由启动类加载器开始尝试从所负责的目录下寻找目标类,如果找到则加载到JVM内存中;
- 如果找不到,则传输到子级加载器,从负责的目录下寻找并加载目标类;
- 依次递归;
- 直到请求的类加载器依旧找不到,则抛出java.lang.ClassNotFoundException异常。





一文教你读懂JVM类加载机制的更多相关文章
- 一文读懂Java类加载机制
Java 类加载机制 Java 类加载机制详解. @pdai Java 类加载机制 类的生命周期 类的加载:查找并加载类的二进制数据 连接 验证:确保被加载的类的正确性 准备:为类的静态变量分配内存, ...
- 一夜搞懂 | JVM 类加载机制
前言 本文已经收录到我的Github个人博客,欢迎大佬们光临寒舍: 我的GIthub博客 学习导图 一.为什么要学习类加载机制? 今天想跟大家唠嗑唠嗑Java的类加载机制,这是Java的一个很重要的创 ...
- 一文教你读懂JVM的类加载机制
Java运行程序又被称为WORA(Write Once Run Anywhere,在任何地方运行只需写入一次),意味着我们程序员小哥哥可以在任何一个系统上开发Java程序,但是却可以在所有系统上畅通运 ...
- 搞懂JVM类加载机制
有这样一道面试题: class Singleton{ private static Singleton singleton = new Singleton(); public static int v ...
- 深入理解JVM虚拟机6:深入理解JVM类加载机制
深入理解JVM类加载机制 简述:虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 下面我们具体 ...
- JVM基础系列第7讲:JVM 类加载机制
当 Java 虚拟机将 Java 源码编译为字节码之后,虚拟机便可以将字节码读取进内存,从而进行解析.运行等整个过程,这个过程我们叫:Java 虚拟机的类加载机制.JVM 虚拟机执行 class 字节 ...
- JVM总结(四):JVM类加载机制
这一节我们来总结一下JVM类加载机制.具体目录如下: 类加载的过程 类加载过程概括 说说引用 详解类加载全过程: 加载 验证 准备 解析 初始化 虚拟机把描述类的数据从Class文件加载到内存,并对数 ...
- JVM 类加载机制详解
如下图所示,JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程. 加载 加载是类加载过程中的一个阶段,这个阶段会在内存中生成一个代表这个类的java.lan ...
- Java虚拟机(四):JVM类加载机制
1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构 ...
随机推荐
- luogu P1712 [NOI2016]区间 贪心 尺取法 线段树 二分
LINK:区间 没想到尺取法. 先说暴力 可以发现答案一定可以转换到端点处 所以在每个端点从小到大扫描线段就能得到答案 复杂度\(n\cdot m\) 再说我的做法 想到了二分 可以进行二分答案 从左 ...
- AGC 043 C - Giant Graph SG函数 dp 贪心
LINK:Giant Graph 神仙题目. 容易发现在图中选择某个点的贡献为\(10^{18\cdot(x+y+z)}\) 这等价于多选一个点多大一点就多乘了一个\(10^{18}\) 所以显然是贪 ...
- MySQL的undo/redo日志和binlog日志,以及2PC
发现自己的知识点有点散,今天就把它们连接起来,好好总结一下. 一.undo log.redo log.binlog的定义和对比 定义和作用 所在架构层级 ...
- [COCOS2DX-LUA]0-006.cocos2dx中关于拖动屏幕物件,同时点击home键,返回后页面变黑的问题。
基本信息介绍: 引擎框架: Quick-Cocos2dx-Community-3.6 测试机型: 魅族MX5 问题简介: 有拖动效果的物件,在拖动的工程中,手指不放,同时点击home键退到后台. 再返 ...
- linux tcpdump抓包Post请求
tcpdump -s 0 -A 'tcp dst port 80 and (tcp[((tcp[12:1] & 0xf0) >> 2):4] = 0x504f5354)' -w f ...
- Elasticsearch第一篇:在 Windows 上的环境搭建
本文介绍如何在 windows 10 ,64位操作系统上安装最新版本 Elasticsearch.以及相关插件.之前看了不少园友的文章,用到的版本都比较低,尤其是插件的版本要和ES的版本相对应等这些问 ...
- C#LeetCode刷题之#48-旋转图像(Rotate Image)
问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/3668 访问. 给定一个 n × n 的二维矩阵表示一个图像. 将 ...
- P、NP、NPC问题详解
转载地址 https://blog.csdn.net/bcb5202/article/details/51202589 P.NP.NPC 概念 > P问题:能够在多项式时间内解决的决策问题. - ...
- jQuery之表单校验:新用户注册
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 洛谷P1036.选数(DFS)
题目描述 已知 n个整数 x1,x2,-,xn,以及11个整数k(k<n).从n个整数中任选k个整数相加,可分别得到一系列的和.例如当n=4,k=3,4个整数分别为3,7,12,19时,可得全部 ...