轻松玩转Java配置的Classpath_JSP编程

  和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中哪一个类先被装载,你必须深入到类装载器的思考方法。事实上,具体实现的时候并没有听起来这么复杂——你只需直接询问类装载器就可以了!

  1: public class JWhich {
  2:
  3: /**
  4: * 根据当前的classpath设置,
  5: * 显示出包含指定类的类文件所在
  6: * 位置的绝对路径
  7: *
  8: * @param className <类的名字>
  9: */
  10: public static void which(String className) {
  11:
  12: if (!className.startsWith("/")) {
  13: className = "/" + className;
  14: }
  15: className = className.replace('.', '/');
  16: className = className + ".class";
  17:
  18: java.net.URL classUrl =
  19: new JWhich().getClass().getResource(className);
  20:
  21: if (classUrl != null) {
  22: System.out.println(" Class '" + className +
  23: "' found in '" + classUrl.getFile() + "'");
  24: } else {
  25: System.out.println(" Class '" + className +
  26: "' not found in '" +
  27: System.getProperty("java.class.path") + "'");
  28: }
  29: }
  30:
  31: public static void main(String args[]) {
  32: if (args.length > 0) {
  33: JWhich.which(args[0]);
  34: } else {
  35: System.err.println("Usage: java JWhich ");
  36: }
  37: }
  38: }

  首先,你必须稍微调整一下类的名字以便类装载器能够接受(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类路径。

时间: 2024-09-16 03:15:34

轻松玩转Java配置的Classpath_JSP编程的相关文章

教你轻松读懂Java中的Socket编程

餐前甜点 Unix的输入输出(IO)系统遵循Open-Read-Write-Close这样的操作范本.当一个用户进程进行IO操作之前,它需要调用Open来指定并获取待操作文件或设备读取或写入的权限.一旦IO操作对象被打开,那么这个用户进程可以对这个对象进行一次或多次的读取或写入操作.Read操作用来从IO操作对象读取数据,并将数据传递给用户进程.Write操作用来将用户进程中的数据传递(写入)到IO操作对象. 当所有的Read和Write操作结束之后,用户进程需要调用Close来通知系统其完成对

金山卫士教你轻松玩转电脑

电脑使用久了,总会对系统方方面面存在或大或小的不满:我习惯使用搜狗拼音而不是QQ拼音,为何我的浏览器首页总是变来变去?我希望用QQ影音而不是暴风影音打开所有电影文件,打开网页如果不再弹出各种广告世界该多清净--如果有个工具能帮我轻松搞定这所有烦恼该多好! 这个可以有.今日小编就为大家精选了15个不得不学的小技巧,能够轻松玩转你的电脑.难得的是这15个小技巧只用一款软件就都搞定了. 金山卫士不但可以查杀目前主流的病毒和木马,修复系统漏洞,优化系统,清理系统垃圾,管理软件,重装系统,还集成了多达29

如何实现JAVA与MATLAB混合编程

问题描述 小弟最近想用JAVA调用MATLAB,但是在配置环境的时候出现了问题.根据网上帖子的要求需要JAVABUILDER这个文件夹,但是我安装了几次MATLAB都没有发现,所以想请教各位高手在这种情况下如何解决JAVA与MATLAB混合编程的问题.求教!!! 解决方案 解决方案二:我以前也遇到这样的问题,根据网上的提示没能解决问题,后来下了2008版的MATLAB里面有那个文件夹,再根据网上的提示就成了.建议你去下一个2008A版解决方案三:我以前也遇到这样的问题,根据网上的提示没能解决问题

玩转Java的CLASSPATH

摘要: =================================== 从表面上看,Java的classpath(类路径)很简单,但一直以来它都是一个产生问题和混乱的根源.本文介绍classpath的基本知识.可能产生的问题,并提供了一个简单的classpath管理工具. =================================== 正文: =================================== 和Java类路径(classpath)打交道的过程中,开发者偶尔会

浅谈Java 8的函数式编程

关于"Java 8为Java带来了函数式编程"已经有了很多讨论,但这句话的真正意义是什么? 本文将讨论函数式,它对一种语言或编程方式意味着什么.在回答"Java 8的函数式编程怎么样"之前,我们先看看Java的演变,特别是它的类型系统,我们将看到Java 8的新特性,特别是Lambda表达式如何改变Java的风景,并提供函数式编程风格的主要优势. 函数式编程语言是什么? 函数式编程语言的核心是它以处理数据的方式处理代码.这意味着函数应该是第一等级(First-cla

微软官方入门教程6:Vista 6步轻松玩转数码照片

本文包含3个部分: 1.6步轻松玩转数码照片: 2.将电脑里的图片添加到照片库 3.升级版照片库带来更多新玩法 6步轻松玩转数码照片 您可以在开始菜单搜索中输入"照片库",就可以找到它了! 1) 导入数码相机的照片 将数码相机与电脑相连,之后 Windows Vista 会自动弹出对话框,选择"导入图片",然后根据对话框的提示,就可以轻松将数码相机里的照片导入到"Windows 照片库".您也可以从开始菜单中的"图片"选项里找

关于java与rs232接口编程的问题

问题描述 关于java与rs232接口编程的问题 最近公司给我个问题,就是用java编写rs232,邀请能读取和发送数据,有没有点高手,给点代码阿,包什么的都弄完了!谢谢各位大神.

缓冲区-java NIO服务器并发编程

问题描述 java NIO服务器并发编程 各位CSDN的编程大神们!求帮忙,求解决方案!跪求! 最近开发一个基于nio的服务器端程序(也包括客户端),实现一个多人(很多人)并发进行即时通讯的东东...发觉这个NIO太难搞了,不过总算还是能够建立一个连接发发"hello world"之类的东西,但是问题来了.由于NIO是针对缓冲区进行操作的,所有数据只能够写入到缓冲区中(我用的是byteBuffer),完了我自定义了一个数据包的格式,如下:|包头一个字节|包长度四个字节|包内容长度可变|

《Java线程与并发编程实践》—— 导读

前言 Java线程与并发编程实践 线程和并发工具并非尤物,但是它们是正式应用的重要部分.本书会向你介绍Java 8 Update 60中线程特性以及并发工具的大部分内容. 第1章介绍了类Thread和接口Runnable.你会学习如何创建Thread以及Runnable对象,获取和设置线程状态.启动线程.中断线程,将一条线程插入另外一条线程以及触发线程睡眠. 第2章关注同步.学习后你会解决一些问题,如没有同步就无法解决的竞态条件.你也能学到如何创建同步方法.块,以及如何使用忽略互斥访问的轻量级同