哪里有 class 告诉我?
说明
本文中的 JVM 参数和代码在 JDK 8 版本生效。
哪里有用户类?
用户类是由开发者和第三方定义的类,它是由应用程序类加载器加载的。
Java 程序可以通过CLASSPATH 环境变量,JVM 启动参数 -cp 或者 -classpath 指定用户需要加载的类的路径。这两个配置的优先级从低到高,后面的配置会覆盖前面的配置,默认值是「.」,即当前路径。

接下来对默认值和优先级进行验证:
验证默认值是当前路径
现在有一个 Temp.java 类,它不在任何包路径下:
public class Temp {
public static void main(String[] args) {
System.out.println("Executed!");
}
}
同时这个时候系统没有配置 CLASSPATH 这个环境变量,如下图所示:

然后拷贝这个编译后的 Temp.class 文件放到 E 盘的下,然后执行命令 java Temp 命令,是能够正常运行这个 Class 文件的。这个时候并没有配置 CLASSPATH 环境变量,同时也没有在执行命令时指定任何参数,说明类加载器是根据 class path 的默认值去找到这个 Class 文件的,这个默认值就是当前路径。如下图所示:

根据官方文档所说 Java 程序启动的时候会把 class path 的值放到 java.class.path 这个系统属性中,如下图所示:

修改上面的代码,在程序运行的时候把实际的 class path 打印出来,代码如下:
public class Temp {
public static void main(String[] args) {
System.out.println("Executed!");
System.out.println("The actual class path is :" + System.getProperty("java.class.path"));
}
}
代码执行结果如下图所示:

可以看到代码打印的结果是「.」,即当前路径。
验证 CLASSPATH 环境变量的作用
增加 Windows 系统环境变量,因为上面是把 Temp.class 文件放到了 E 盘下面,所以这里设置的 CLASSPATH 环境变量也是 E 盘,如下图所示:


再次运行程序,执行结果如下图所示:

程序能够正常执行说明通过配置的 E: 这个路径,类加载器能够找到 Temp.class 文件。同时打印的 class path 也是 E: ,符合设置。
验证 -cp 或者 -classpath 参数的作用
把上面设置的 CLASSPATH 环境变量删除,然后通过执行 java 命令的时候指定 -cp 参数来设置 class path 的路径。如下图所示:


程序执行的效果和通过 CLASSPATH 环境变量设置的相同。
验证 -cp或 -classpath 参数的优先级高于 CLASSPATH 环境变量
设置 CLASSPATH 环境变量为 D: ,如下图所示:


如果不带 -cp 参数执行执行会提示找不到类,因为 D: 路径下没有 Temp.class 这个文件。如下图所示:

带上 -cp 参数后就能够正常执行,这个时候两个配置都有,但是 -cp 参数的配置生效了,说明 -cp 参数的优先级高于 CLASSPAHTH 环境变量,如下图所示:

哪里有引导类?
sun.boot.class.path 系统属性的值
引导类指的是构成 Java 平台的类,包括 rt.jar 中的类以及其他几个重要的 jar 文件中的类,它们是由引导类加载器(Bootstrap ClassLoader)加载的。
在前面可以看到如果直接在 Temp.class 文件所在的路径下执行 java Temp命令就能够正常执行。那这个 Temp 类的父类是 Object 类,这个类是在 jre/lib 目录下的 rt.jar 包中,但是没有任何地方指定了这个路径,那引导类加载器(BootstrapClassLoader) 是如何找 Object 类并加载的呢?
根据官方文档说的引导类加载器加载的 class path 可以通过 sun.boot.class.path 这个系统属性获取到,如下图所示:

把上面的代码修改为如下:
public class Temp {
public static void main(String[] args) {
System.out.println("Executed!");
System.out.println("The actual class path is :" + System.getProperty("sun.boot.class.path"));
}
}
通过 java Temp 命令执行后(不配置 CLASSPATH 环境变量,让它使用默认值),结果如下:

可以看到输出结果里面有 rt.jar 包的绝对路径。实际上并没有任何地方指定了这个路径,那么这个路径怎么获取到并设置到 sun.boot.class.path 这个系统属性中的呢?
sun.boot.class.path 系统属性赋值源码分析
这里以 Windows 平台为例分析一下 HotSpot 虚拟机的源码实现。这里主要涉及到三个文件的内容,分别是:
hotspot\src\share\vm\runtime\arguments.cpp
hotspot\src\share\vm\runtime\os.cpp
hotspot\src\os\windows\vm\os_windows.cpp

源代码的调用链路如下:

arguments.cpp 负责处理 JVM 启动的参数,在这个文件中会初始化 _java_home 和_sun_boot_class_path 系统属性,如下图所示:

然后调用 os_windows.cpp 的 init_system_properties_values() 方法,在该方法中又会调用 os_windows.cpp 中的 jvm_path() 方法,该方法中会尝试去获取 jvm.dll 的绝对路径,如下图所示:



然后返回到os_windows.cpp 的 init_system_properties_values() 方法,去除掉路径中的 jvm.dll,server/client,bin 然后放入到前面创建的 _java_home 系统属性中,如下图所示:

然后再继续调用 os.cpp 中的 set_boot_path() 方法,在这个方法中获取 _java_home 系统属性中的值,用来格式化引导类 jar 包的路径,然后放入到 _sun_boot_class_path 中。 如下图所示:

这就是 sun.boot.class.path 系统属性值在 Java 程序启动时的设置过程。
深入理解Java虚拟机中介绍到「引导类加载器负责加载存放在<JAVA_HOME>/lib 目录下或者被 -Xbootclasspath 参数所指定的路径中存放的,且是 Java 虚拟机能够识别的(按照文件名识别,例如 rt.jar,名字不符合的类库即使是放在 lib 目录中也不会被加载)」。这里所描述的「按照文件名识别」指的应该就是上面 os.cpp 的 set_boot_path() 方法中定义的路径常量,只有这些路径常量才会被格式化最终放到 sun.boot.class.path 系统属性中。
目前这个系统属性在 JDK 9 中已经被移除了,如下图所示:

引导类的路径可以通过 sun.boot.class.path 系统属性或者 -Xbootclasspath JVM 参数设置。
神奇的 -Xbootclasspath/p 参数
除此之外 JVM 还提供了两个参数 -Xbootclasspath/p 和 -Xbootclasspath/a,分别用于在默认的引导类路径前面和后面增加所配置的路径。如下图所示:


-Xbootclasspath/p 这个参数有点意思,它可以用来修复引导类的 Bug 或者扩展类的功能。
比如现在把 java.util.Collections 类拷贝出来,给它增加一个方法 extendMethod(),然后打包成 jar 包,如下图所示:


在代码中通过反射的方式调用 extendMethod() 方法,代码如下所示:
public class Temp {
public static void main(String[] args) throws Exception {
Method method = Collections.class.getDeclaredMethod("extendMethod");
method.invoke(null);
}
}
在执行 java 命令时通过 -Xbootclasspath/p 配置上这个 jar 包。可以看到新增的方法被成功调用了,说明 extend.jar 包中的 Collections 类覆盖了默认的 java.util.Collections 类,因为它在所有的路径前面,所以先被类加载器加载。如下图所示:


这个参数在 JDK 9 中也被移除了,取而代之的是 --patch-module 参数,如下图所示:

参考
findingclasses
PATH and CLASSPATH
JDK 9 Release Notes
JEP 261
How can we overwrite java.base/java.lang.Integer from OpenJDK 11 using --patch-module?
深入理解Java虚拟机
哪里有 class 告诉我?的更多相关文章
- 【极品代码】一般人我不告诉他,手机端h5播放时不自动全屏代码
已测适用于ios,某些安卓手机微信下播放视频会出现播放器控件(这个实在是无力吐槽噢,因为之前还遇到过微信播放完视频后竟然无法退出全屏出现广告的情况,只有播放完后刷新页面并且要放到框架页里才能屏蔽微信视 ...
- 一张图告诉你,只会HTML还不够!
会了HTML和HTML5语法,你就真的会了HTML吗,来看这张图!是这本<超实用的HTML代码段>入门实例书的导览!熊孩子们,赶紧学习去吧! 如果一半以上的你都不会,必须看这本书,阿里一线 ...
- 万能的林萧说:我来告诉你,一个草根程序员如何进入BAT。
引言 首先声明,不要再问LZ谁是林萧,林萧就是某著名程序员小说的主角名字. 写这篇文章的目的其实很简单,算是对之前LZ一篇文章的补充和完善. 之前LZ写过一篇<回答阿里社招面试如何准备,顺便谈谈 ...
- #CSDN刷票门# 有没有人在恶意刷票?CSDN请告诉我!用24小时监控数据说话!
特别声明: 此次并非针对其他参与2013中国十大优秀开源项目的同行,体系有漏洞要谴责的是制定规则并从中获益但不作为的权贵,草根们制定不了规则但可发现和利用漏洞,这是程序员应有反叛精神没错.但被作为道具 ...
- [猜数字]把两个数和告诉A,积告诉B,求这两个数是什么
1-20的两个数把和告诉A,积告诉B,A说不知道是多少,B也说不知道,这时A说我知道了,B接着说我也知道了,问这两个数是多少? 分析: 设和为S,积为M. 首先,A:我不知道. 说明:S可以分解成多个 ...
- 很少有人会告诉你的Android开发基本常识
原文:很少有人会告诉你的Android开发基本常识. 文章介绍了一些关于开发.测试.版本管理.工具使用等方面的知识.
- 程序最多能new多少内存(2G内存里要放程序的5大区,HeapAlloc比new要快多了,而且超过2G的时候会告诉你)
根据<Windows核心编程>得知:X86操作系统提供每个程序最多只有4G的虚拟内存,其中2G虚拟内存提供给系统用(具体用来干什么还待考察),还有2G的内存留给用户使用.那这2G内存能拿来 ...
- 告诉你吧,一套皮肤在winform与wpf开发模式下实现的界面效果同样精彩,winform界面和wpf界面。
一.同一资源: 二.先上软件界面: (1)wpf界面: 在wpf中实现这样类似web风格的软件界面就不用我多说了,在wpf实现这样的风格是很简单的,完全像网页设计一样的. (2)winform界面 在 ...
- 一篇文章告诉你为何GitHub估值能达20亿美元
软件开发平台GitHub今日宣布,已获得硅谷多家知名风投2.5亿美元融资,这也让其融资总额达到了3.5亿美元,此轮融资对GitHub的估值约为20亿美元. GitHub有何特别之处? GitHub创立 ...
- 大数据:从开源告诉你身边的IT故事
最近我们Team利用Dream分布式计算平台,做了这样一件事情,将Github的大量数据通过爬虫抓取下来,通过分析后,我们抽取最近一年中部分的开发者和项目信息,得到了如下有趣的信息,故分享之,数据原汁 ...
随机推荐
- v-if 为什么不能和 v-for 一起使用 ?
当 Vue 处理指令时,v-for 比 v-if 具有更高的优先级,通过v-if 移动到容器元素,不会再重复遍历列表中的每个值.取而代之的是,我们只检查它一次,且不会在 v-if 为否的时候运算 v- ...
- element 的 表单 搜索 和 重置按钮换行问题 ? inline
想要: 但是: 加上
- apache安装详解
Apache安装 准备工作. 首先在C盘根目录下创建一个名为web的文件夹作为php开发环境的安装位置,并在web文件夹中创建apache24子文件夹,将apache解压后文件放至此处. 安装包 首先 ...
- 三大主流负载均衡软件对比(LVS+Nginx+HAproxy)
LVS: 优点 : 1.抗负载能力强.性能高,能达到F5硬件的60%:对内存和cpu资源消耗比较低2.工作在网络4层,通过vrrp协议转发(仅作分发之用),具体的流量由linux内核处理,因此没有流量 ...
- Android添加OpenCV支持
首先下载OpenCV的SDK 推荐在官网下载. 官网地址:https://opencv.org/releases/ 也可以在OpenCV的GitHub上现在 GitHUb链接:https://gith ...
- Golang 开源库分享:anko - 给 Go 加点“脚本魔法”
GitHub 仓库链接:https://github.com/mattn/anko 1. anko 是干嘛用的? anko 是一个可以让 Go 项目支持脚本语言的小工具.换句话说,就是我们可以给 Go ...
- 矩阵怪 - 2024全新矩阵产品,一键分发抖音,快手,视频号,B站,小红书!
本方案面向谁,解决了什么问题 本方案主要面向C端客户,特别是那些在各大短视频平台(如小红书.抖音.视频号.快手.B站等)上进行内容创作和分发的个人用户.自由职业者.小型团队或企业.这些用户通常面临着在 ...
- 数据结构之链表篇(单链表,循环链表,双向链表)C语言版
1.链表 链表是线性表的一种,由一系列节点(结点)组成,每个节点包含一个数据域和一个指向下一个节点的指针域.链表结构可以克服数组需要预先知道数据大小的缺点,而且插入和删除元素很方便,但是失去数组随机读 ...
- js+jquery实现贪吃蛇经典小游戏
项目只使用到了html,css,js,jquery技术点,没有使用游戏框架,下载本地直接双击index.html 运行即可体验游戏效果. 项目展示 进入游戏 游戏开始 游戏暂停 html文件 < ...
- 用java实现JDBC数据库连接池
这次写数据库连接池主要想解决的还是servlet访问数据库获取数据的稳定性问题,于是便研究了一下,下面来讲一讲如何用java来写一个适合自己用的数据库连接池.这个东西大家并不陌生,通过使用数据连接池我 ...