广告
返回顶部
首页 > 资讯 > 精选 >如何深入探讨、理解Java的CLASSPATH
  • 248
分享到

如何深入探讨、理解Java的CLASSPATH

2023-06-03 02:06:22 248人浏览 八月长安
摘要

这篇文章给大家介绍如何深入探讨、理解Java的CLASSPATH,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。从表面上看,Java的classpath(类路径)很简单,但一直以来它都是一个产生问题和混乱的根源。本文介绍

这篇文章给大家介绍如何深入探讨、理解Java的CLASSPATH,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。

从表面上看,Java的classpath(类路径)很简单,但一直以来它都是一个产生问题和混乱的根源。本文介绍classpath的基本知识、可能产生的问题,并提供了一个简单的classpath管理工具

和Java类路径(classpath)打交道的过程中,开发者偶尔会遇到麻烦。这是因为,类装载器实际装入的是哪一个类有时并不显而易见,当应用程序的classpath包含大量的类和目录时,情况尤其严重。本文将提供一个工具,它能够显示出被装入类文件的绝对路径名。

一、Classpath基础

Java虚拟机JVM)借助类装载器装入应用程序使用的类,具体装入哪些类根据当时的需要决定。CLASSPATH环境变量告诉类装载器到哪里去寻找第三方提供的类和用户定义的类。另外,你也可以使用JVM命令行参数-classpath分别为应用程序指定类路径,在-classpath中指定的类路径覆盖CLASSPATH环境变量中指定的值。

类路径中的内容可以是:文件的目录(包含不在包里面的类),包的根目录(包含已打包的类),包含类的档案文件(比如.zip文件或者.jar文件)。在Unix家族的系统上,类路径的各个项目由冒号分隔,在MS windows系统上,它们由分号分隔。

类装载器以委托层次的形式组织,每一个类装载器有一个父类装载器。当一个类装载器被要求装载某个类时,它在尝试自己寻找类之前会把请求先委托给它的父类装载器。系统类装载器,即由安装在系统上的jdk或JRE提供的默认类装载器,通过CLASSPATH环境变量或者-classpath这个JVM命令行参数装入第三方提供的类或者用户定义的类。系统类装载器委托扩展类装载器装入使用Java Extension机制的类。扩展类装载器委托自举类装载器(bootstrap class loader)装入核心JDK类。

你可以自己开发特殊的类装载器,定制JVM如何动态地装入类。例如,大多数Servlet引擎使用定制的类装载器,动态地装入那些在classpath指定的目录内发生变化的类。

必须特别注意的是(也是令人吃惊的是),类装载器装入类的次序就是类在classpath中出现的次序。类装载器从classpath的第一项开始,依次检查每一个设定的目录和压缩文件,尝试找出待装入的类文件。当类装载器第一次找到具有指定名字的类时,它就把该类装入,classpath中所有余下的项目都被忽略。

看起来很简单,对吧?

二、可能出现的问题

不管他们是否愿意承认,初学者和富有经验的Java开发者都一样,他们都曾经在某些时候(通常是在那些最糟糕的情形下)被冗长、复杂的classpath欺骗。应用程序所依赖的第三方类和用户定义类的数量逐渐增长,classpath也逐渐成了一个堆积所有可能的目录和档案文件名的地方。此时,类装载器首先装载的究竟是哪一个类也就不再显而易见。如果classpath中包含重复的类入口,这个问题尤其突出。前面已经提到,类装载器总是装载第一个它在classpath中找到的具有合适名字的类,从实际效果看,它“隐藏”了其他具有合适名字但在classpath中优先级较低的类。

如果不小心,你很容易掉进这个classpath的陷阱。当你结束了一天漫长的工作,最后为了让应用程序使用最好、最新的类,你把一个目录加入到了classpath,但与此同时,你却忘记了:在classpath的另一个具有更高优先级的目录下,存放着该类的另一个版本!

三、一个简单的classpath工具

优先级问题是扁平路径声明方法与生俱来固有的问题,但它不是只有Java的classpath才有的问题。要解决这个问题,你只需站到富有传奇色彩的软件巨构的肩膀上:Unix操作系统有一个which命令,在命令参数中指定一个名字,which就会显示出当这个名字作为命令执行时执行文件的路径名。实际上,which命令是分析PATH变量,然后找出命令第一次出现的位置。对于Java的类路径管理来说,这应该也是一个好工具。在它的启发之下,我着手设计了一个Java工具JWhich。这个工具要求指定一个Java类的名字,然后根据classpath的指引,找出类装载器即将装载的类所在位置的绝对路径。

下面是一个JWhich的使用实例。它显示出当Java类装载器装载com.clarkware.ejb.ShoppinGCartBean类时,该类第一次出现位置的绝对路径名,查找结果显示该类在某个目录下:

> java JWhich com.clarkware.ejb.ShoppingCartBean

Class 'com.clarkware.ejb.ShoppingCartBean' found in '/home/mclark/classes/com/clarkware/ejb/ShoppingCartBean.class'

下面是第二个JWhich的使用实例。它显示出当Java类装载器装载javax.servlet.Http.httpservlet类时,该类第一次出现位置的绝对路径名,查找结果显示该类在某个档案文件中:

> java JWhich javax.servlet.http.HttpServlet

Class 'javax.servlet.http.HttpServlet' found in 'file:/home/mclark/lib/servlet.jar!/javax/servlet/http/HttpServlet.class'

四、JWhich的工作过程

要精确地测定classpath中哪一个类先被装载,你必须深入到类装载器的思考方法。事实上,具体实现的时候并没有听起来这么复杂——你只需直接询问类装载器就可以了!

public class JWhich {

public static void which(String className) {

if (!className.startsWith("/")) {

className = "/" + className;

}

className = className.replace('.', '/');

className = className + ".class";

java.net.URL classUrl =

new JWhich().getClass().getResource(className);

if (classUrl != null) {

System.out.println("nClass '" + className +

"' found in n'" + classUrl.getFile() + "'");

} else {

System.out.println("nClass '" + className +

"' not found in n'" +

System.getProperty("java.class.path") + "'");

}

}

public static void main(String args[]) {

if (args.length > 0) {

JWhich.which(args[0]);

} else {

System.err.println("Usage: java JWhich");

}

}

}

首先,你必须稍微调整一下类的名字以便类装载器能够接受(12-16行)。在类的名字前面加上一个“/”表示要求类装载器对classpath中的类名字进行逐字精确匹配,而不是尝试隐含地加上调用类的包名字前缀。把所有“.”转换为“/”的目的是,按照类装载器的要求,把类名字格式化成一个合法的URL资源名。

接下来,程序向类装载器查询资源,这个资源的名字必须和经过适当格式化的类名字匹配(18-19行)。每一个Class对象维护着一个对装载它的ClassLoader对象的引用,所以这里是向装载JWhich类的类装载器查询。Class.getResource()方法实际上委托装入该类的类装载器,返回一个用于读取类文件资源的URL;或者,当指定的类名字不能在当前的classpath中找到时,Class.getResource()方法返回null。

最后,如果当前的classpath中能够找到指定的类,则程序显示包含该类的类文件所在位置的绝对路径名(21-24行)。作为一种调试辅助手段,如果当前classpath中不能找到指定的类,则程序获取java.class.path系统属性并显示当前的classpath(24-28行)。

很容易想象,在使用Servlet引擎classpath的Java Servlet中,或者在使用EJB服务器classpath的EJB组件中,上面这段简单的代码是如何运作。例如,如果JWhich类是由Servlet引擎的定制类装载器装入,那么程序将用Servlet引擎的类装载器去寻找指定的类。如果Servlet引擎的类装载器不能找到类文件,它将委托它的父类装载器。一般地,当JWhich被某个类装载器装入时,它能够找出当前类装载器以及所有其父类装载器所装入的所有类。

如果需要是所有发明之母,那么帮助我们管理Java类路径的工具可以说迟到了很长时间。Java新闻组和邮件列表中充塞着许多有关classpath的问题,现在JWhich为我们提供了一个简单却强大的工具,帮助我们在任何环境中彻底玩转Java类路径。

关于如何深入探讨、理解Java的CLASSPATH就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

--结束END--

本文标题: 如何深入探讨、理解Java的CLASSPATH

本文链接: https://www.lsjlt.com/news/232163.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 如何深入探讨、理解Java的CLASSPATH
    这篇文章给大家介绍如何深入探讨、理解Java的CLASSPATH,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。从表面上看,Java的classpath(类路径)很简单,但一直以来它都是一个产生问题和混乱的根源。本文介绍...
    99+
    2023-06-03
  • Java HashMap工作原理该如何深入探讨
    本篇文章为大家展示了Java HashMap工作原理该如何深入探讨,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。大部分Java开发者都在使用Map,特别是HashMap。HashMap是一种简单但强...
    99+
    2023-06-17
  • 深入探讨JAVA中的异常与错误处理
    异常与错误:  异常:  在Java中程序的错误主要是语法错误和语义错误,一个程序在编译和运行时出现的错误我们统一称之为异常,它是VM(虚拟机)通知你的一种方式,通过这种方式,VM让...
    99+
    2022-11-15
    JAVA 异常 错误处理
  • 深入探讨JavaScript中的内存管理
    内存管理是编程语言的基本能力,JavaScript 中的内存管理是通过 V8 完成的。V8 的实现遵循 ECMA-262 规范,而规范中没有阐述内存布局以及内存管理相关信息,所以它的原理取决于解释器的实现。唯一肯定的是不管任何编程语言,内存...
    99+
    2023-05-14
    前端 JavaScript
  • 怎样实现Java线程的深入探讨
    这篇文章将为大家详细讲解有关怎样实现Java线程的深入探讨,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.什么是线程    一般来说,我们把正在计算机中执行的程序叫做&...
    99+
    2023-06-03
  • 深入探讨Golang中的数据处理方法
    深入探讨Golang中的数据处理方法导言:在当今的信息时代,数据是无处不在的。对于开发者来说,处理数据是日常工作中不可或缺的一部分。而Go语言(Golang)作为一种强大且高效的编程语言,提供了许多灵活且易用的数据处理方法。本文将深入探讨G...
    99+
    2023-12-23
    方法 Golang 数据处理
  • 深入探讨java后端和PHP后端的区别
    随着互联网的不断发展,软件开发领域也在不断地进步和发展。作为后端开发两个重要的编程语言,Java和PHP在很多方面都有着不同的表现和适用场景。本篇文章将从技术层面深入探讨Java后端和PHP后端的区别。一、语言类型及堆栈Java是一种静态面...
    99+
    2023-05-14
    php java
  • 如何进行Java多线程同步机制的深入探讨
    今天就跟大家聊聊有关如何进行Java多线程同步机制的深入探讨,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。在很多程序员的脑海中Java多线程都会占据一定的位置,众所周知,在Java多...
    99+
    2023-06-17
  • 深入探讨Java接口和容器:如何更好地管理文件?
    Java作为一种面向对象编程语言,拥有着非常强大的接口和容器的功能。接口是一种特殊的类,它定义了一组抽象方法,不包含任何实现。容器是一种可以存储多个对象的数据结构,Java中最常用的容器包括List、Set、Map等。在Java中,接口和容...
    99+
    2023-11-04
    接口 容器 文件
  • 深入探讨vite是怎么解析.env文件的
    使用vue框架开发前端项目时,我们部署的时候都会部署多套环境,往往开发、测试以及线上环境调用的接口域名都是不一样的。如何能做到区分呢?那就是使用环境变量和模式。使用vite构建的vue3项目中,可以在根目录下创建.env.[模式]文件定义一...
    99+
    2023-05-14
    前端 Vite Vue.js
  • 怎样进行Java线程控制权源代码的深入探讨
    怎样进行Java线程控制权源代码的深入探讨,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Java线程控制权的问题十分重要,我们在不断的学习中需要不断的注意相关的问题。下面我们就...
    99+
    2023-06-17
  • 深入探讨z-index属性及其常见属性值:理解绝对定位
    理解绝对定位的常用属性值:深入解析CSS中的z-index属性在CSS中,绝对定位(absolute positioning)是一种常用的定位方式,用于精确控制元素在页面中的位置。而其中的一个重要属性值,z-index,可以帮助我们决定元素...
    99+
    2023-12-28
    z-index 绝对定位 (Absolute Positioning) 定位 (Positioning)
  • java String的深入理解
    java String的深入理解一、Java内存模型 按照官方的说法:Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。     JVM主要管理两种类型内存...
    99+
    2023-05-31
    java string ava
  • 如何深入理解Java中的接口
    今天就跟大家聊聊有关如何深入理解Java中的接口,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。一、前言前面我们说了抽象类的概述,我们对抽象类也有个认识和理解了,现在我们学习十分重要的...
    99+
    2023-06-21
  • 深入理解Java中的HashMap
    目录一、HashMap的结构图示二、HashMap的成员变量以及含义2.1、hash方法说明2.2、tableSizeFor方法说明三、HashMap的构造方法四、HashMap元素...
    99+
    2022-11-12
  • 深入探讨Go中对象和函数的应用:如何构建高质量的框架?
    在Go语言中,对象和函数是构建高质量框架的重要组成部分。在本文中,我们将深入探讨Go中对象和函数的应用,以及如何使用它们来构建高质量的框架。 对象是指具有属性和方法的数据类型,它们可以封装相关的数据和操作。在Go中,对象被称为结构体。结构体...
    99+
    2023-11-03
    对象 函数 框架
  • 如何理解enq: TX - index contention等待的探讨与测试
    本篇文章给大家分享的是有关如何理解enq: TX - index contention等待的探讨与测试,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看...
    99+
    2022-10-19
  • 深入探究如何使用Java编写MapReduce程序
    目录MapReduce的原理Map阶段Reduce阶段Shuffle阶段MapReduce程序实现总结 MapReduce的原理 MapReduce由两个主要阶段组成:Map和Red...
    99+
    2023-05-19
    Java编写MapReduce程序 Java编写MapReduce Java MapReduce
  • 如何深入Java核心探秘Java垃圾回收机制
    这期内容当中小编将会给大家带来有关如何深入Java核心探秘Java垃圾回收机制,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。垃圾收集GC(Garbage Collection)是Java语言的核心技术之一...
    99+
    2023-06-17
  • 深入探究Java线程不安全的原因与解决
    目录一、什么是线程安全二、线程不安全的原因1、修改共享数据2、原子性3、内存可见性4、指令重排序三、解决线程安全方案一、什么是线程安全 想给出一个线程安全的确切定义是复杂的,但我们可...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作