哪里有 class 告诉我?

news/2024/12/25 20:09:28/文章来源:https://www.cnblogs.com/javadaydayup/p/18631330

说明

本文中的 JVM 参数和代码在 JDK 8 版本生效。

哪里有用户类?

用户类是由开发者和第三方定义的类,它是由应用程序类加载器加载的。

Java 程序可以通过CLASSPATH 环境变量,JVM 启动参数 -cp 或者 -classpath 指定用户需要加载的类的路径。这两个配置的优先级从低到高,后面的配置会覆盖前面的配置,默认值是「.」,即当前路径。

image.png

接下来对默认值和优先级进行验证:

验证默认值是当前路径

现在有一个 Temp.java 类,它不在任何包路径下:

public class Temp {  public static void main(String[] args) {  System.out.println("Executed!");  }  
}

同时这个时候系统没有配置 CLASSPATH 这个环境变量,如下图所示:

image.png

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

image.png

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

image.png

修改上面的代码,在程序运行的时候把实际的 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"));  }  
}

代码执行结果如下图所示:
image.png

可以看到代码打印的结果是「.」,即当前路径。

验证 CLASSPATH 环境变量的作用

增加 Windows 系统环境变量,因为上面是把 Temp.class 文件放到了 E 盘下面,所以这里设置的 CLASSPATH 环境变量也是 E 盘,如下图所示:

image.png

image.png

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

程序能够正常执行说明通过配置的 E: 这个路径,类加载器能够找到 Temp.class 文件。同时打印的 class path 也是 E: ,符合设置。

验证 -cp 或者 -classpath 参数的作用

把上面设置的 CLASSPATH 环境变量删除,然后通过执行 java 命令的时候指定 -cp 参数来设置 class path 的路径。如下图所示:

image.png

image.png

程序执行的效果和通过 CLASSPATH 环境变量设置的相同。

验证 -cp-classpath 参数的优先级高于 CLASSPATH 环境变量

设置 CLASSPATH 环境变量为 D: ,如下图所示:

image.png

image.png

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

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

哪里有引导类?

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 这个系统属性获取到,如下图所示:
image.png

把上面的代码修改为如下:

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 环境变量,让它使用默认值),结果如下:

image.png

可以看到输出结果里面有 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

image.png

源代码的调用链路如下:

arguments.jpg

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

image.png

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

image.png

image.png
image.png

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

image.png

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

image.png

这就是 sun.boot.class.path 系统属性值在 Java 程序启动时的设置过程。

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

目前这个系统属性在 JDK 9 中已经被移除了,如下图所示:

image.png

引导类的路径可以通过 sun.boot.class.path 系统属性或者 -Xbootclasspath JVM 参数设置。

神奇的 -Xbootclasspath/p 参数

除此之外 JVM 还提供了两个参数 -Xbootclasspath/p-Xbootclasspath/a,分别用于在默认的引导类路径前面和后面增加所配置的路径。如下图所示:

image.png

image.png

-Xbootclasspath/p 这个参数有点意思,它可以用来修复引导类的 Bug 或者扩展类的功能。

比如现在把 java.util.Collections 类拷贝出来,给它增加一个方法 extendMethod(),然后打包成 jar 包,如下图所示:

image.png
image.png

在代码中通过反射的方式调用 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 类,因为它在所有的路径前面,所以先被类加载器加载。如下图所示:

image.png

image.png

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

image.png

参考

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虚拟机

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/858804.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python多进程之间通讯,消息队列Queue

代码:from multiprocessing import Process, Queuedef producer(q):myinfo = "包子"q.put(myinfo)print(f"生产了{myinfo}")myinfo = "饺子"q.put(myinfo)print(f"生产了{myinfo}\n") 生产了4个,消费5个def consumer(q):print(f&q…

使用DBeaver连接带有Kerberos认证的hive(亲测可用)

先下载工具 https://yvioo.lanzn.com/isBg42j0fu7e里面是两个文件 一个jar包 一个安装包 首先点击kfw-4.1-amd64.msi 进行安装,建议直接默认配置安装 选择"TYPE" 安装完成后 点击 1、先配置环境变量 第一个变量名:KRB5_CONFIG 变量值: 这个就是Kerberos认证给的k…

【QTTabBar】批量去除当前文件夹的所有文件只读属性

使用方法参考: https://www.cnblogs.com/issacnew/p/18392262// 作者:博客园-issacnew // 网站:https://www.cnblogs.com/issacnew/p/18392262 // 作用:qttabbar去除当前文件夹下的所有文件只读属性,使得所有文件可读var qs = new ActiveXObject("QTTabBarLib.Script…

C#使用Python.NET执行Python脚本文件踩坑总结

在VS,Nuget包管理器搜索“Python.NET”,安装pythonnet包,如下图: C#使用Python.NET执行Python脚本文件,C#代码如下:1 public class PythonExecuter2 {3 private readonly string _pythonDllPath;4 private readonly string _workDir;5 6 public PythonExecut…

PoerPC平台下的ethtool工具的编译

1. 编译器安装 2. 编译过程 参考:ethtool工具源码交叉编译_ethtool交叉编译-CSDN博客 注:步骤3中的config配置中,通过sudo apt-get install pkg-config libmnl-dev修复完问题后,需要重新执行一次步骤2中的./autogen.sh下载目标程序

微信小程序开发总结

业务需要,最近又搞起了微信小程序,之前从来没有参与过小程序的开发,对于开发中的流程也是知之甚少,正好学习一下,开搞... 前提:使用企业注册小程序 微信认证 小程序备案 [本地开发] 1.获取appid和secret 管理 > 开发管理 获取即可, 需要管理员扫码确认这里获取到的appid在使…

windows 下面使用 celery 管理定时任务

Python 实现定时任务有以下几种思路使用子进程(现成)+ time.sleep 间隔执行 使用现有的库管理定时任务如,celery, tornado等 使用系统的机制执行linux 下面 crontab ,windows 下面taskschd.msc本次调查 celery 这个常用的异步任务管理框架,它有一下好处支持分布式 支持任…

Goby 漏洞发布|CVE-2024-9047 WordPress File Upload 插件 wfu_file_downloader.php 任意文件读取漏洞

漏洞名称:CVE-2024-9047 WordPress File Upload 插件 wfu_file_downloader.php 任意文件读取漏洞 English Name:CVE-2024-9047 WordPress File Upload Plugin wfu_file_downloader.php Arbitrary File Read Vulnerabilit CVSS core: 6.8 漏洞描述: WordPress File Upload插件…

002. 队列安排(洛谷P1160)

002. 队列安排(洛谷P1160) 题目描述 一个学校里老师要将班上 \(N\) 个同学排成一列,同学被编号为 \(1\sim N\),他采取如下的方法:先将 \(1\) 号同学安排进队列,这时队列中只有他一个人;\(2\sim N\) 号同学依次入列,编号为 \(i\) 的同学入列方式为:老师指定编号为 \(i\) …

一个GLSL Shader的格式化算法(LALR解析器)

一个GLSL Shader的格式化算法(LALR解析器) 在进行OpenGL程序开发时,我需要自行解析`string`类型的Shader代码,抽取出里面的某些变量名和subroutine名。 由于找不到可用的GLSL Shader解析器,就照着虎书(《现代编译原理-c语言描述》)自己写了个LALR Generator,实际上包含了…

adb使用教程

谷歌官方出品用来控制安卓手机的工具1、作用打印日志定位bug稳定性测试运行设备的shell命令上传和下载文件安装和卸载设备上的应用等2、adb的安装配置Android开发官网下载ADB压缩包解压压缩包将ADB包放到根目录下将ADB路径加入到环境变量里3、用adb连接手机 进入开发者模式USB连…