这个四部分构成的文章系列研究 java™ 的类装入问题,帮助应用程序开发人员理解和调试可能碰到的问题。在第 2 部分中,来自 IBM Hursley 实验室的作者 Lakshmi Shankar 和 Simon Burns 处理的一些异常,虽然相当简单,但是却经常困扰着新老 Java 开发人员。
本文是系列中四篇文章的第二篇,它考察了一般在运行应用程序时抛出的各种类装入异常。这些异常虽然常见,但是 Java 程序员对它们的理解通常不深。本文将逐个提供能够突出这些异常行为特点的具体示例,解释造成异常的原因,并介绍一些可行的解决技术。文章从非经常见的 ClassNotFoundException
开始,逐渐转移到不太常见的异常,例如 ExceptionInInitializerError
。
在开始这篇文章之前,应当熟悉类装入委托模型,以及类链接的阶段和过程。我们强烈建议您从阅读这个系列的 第一篇文章 开始。
ClassNotFoundException
ClassNotFoundException
是最常见的类装入异常类型。它发生在装入阶段。Java 规范对 ClassNotFoundException
的描述是这样的:
当应用程序试图通过类的字符串名称,使用以下三种方法装入类,但却找不到指定名称的类定义时抛出该异常。
Class
中的 forName()
方法。 ClassLoader
中的 findSystemClass()
方法。 ClassLoader
中的 loadClass()
方法。所以,假如显式地装入类的尝试失败,那么就抛出 ClassNotFoundException
。清单 1 中的测试用例提供的示例代码抛出了一个 ClassNotFoundException
:
import java.net.MalformedURLException;import java.net.URL;import java.net.URLClassLoader;public class ClassNotFoundExceptionTest { public static void main(String args[]) { try { URLClassLoader loader = new URLClassLoader(new URL[] { new URL( "file://C:/CL_Article/ClassNotFoundException/")}); loader.loadClass("DoesNotExist"); } catch (ClassNotFoundException e) { e.PRintStackTrace(); } catch (MalformedURLException e) { e.printStackTrace(); } }}
这个测试用例定义了一个类装入器(MyClassLoader
),用于装入一个不存在的类(DoesNotExist
)。当它运行时,会出现以下异常:
java.lang.ClassNotFoundException: DoesNotExist at java.net.URLClassLoader.findClass(URLClassLoader.java:376) at java.lang.ClassLoader.loadClass(ClassLoader.java:572) at java.lang.ClassLoader.loadClass(ClassLoader.java:504) at ClassNotFoundExceptionTest.main(ClassNotFoundExceptionTest.java:11)
因为这个测试试图使用对 loadClass()
的显式调用来进行装入,所以抛出 ClassNotFoundException
。
通过抛出 ClassNotFoundException
,类装入器提示,定义类时所需要的字节码在类装入器所查找的位置上不存在。这些异常修复起来通常比较简单。可以用 IBM 的 verbose 选项检查类路径,确保使用的类路径设置正确(要获得 verbose 的更多信息,请参阅本系列的 第一篇文章)。假如类路径设置正确,但是仍然看到这个错误,那么就是需要的类在类路径中不存在。要修复这个问题,可以把类移动到类路径中指定的目录或 JAR 文件中,或者把类所在的位置添加到类路径中。
NoClassDefFoundError
NoClassDefFoundError
是类装入器在装入阶段抛出的另一个常见异常。JVM 规范对 NoClassDefFoundError
的定义如下:
假如 Java 虚拟机或ClassLoader
实例试图装入类定义(作为正常的方法调用的一部分,或者作为使用 new 表达式创建新实例的一部分),但却没有找到类定义时抛出该异常。
当目前执行的类已经编译,但是找不到它的定义时,会存在 searched-for 类定义。
新闻热点
疑难解答