在执行 Java程序时经常会碰到ClassNotFoundException和NoClassDefFoundError两个异常,它们都和类加载有关,下面详细分一下这两个异常的原因。
ClassNotFoundException恐怕时Java程序员经常碰到的异常,尤其是初学者来说,简直是让人崩溃,明明那个类就在那里,为啥就是找不到呢?无数个Java程序员都是这样问过自己。 这个异常通常发生在显式加载类的时候,例如用如下方式调用加载一个类时就报这个错了。
public class NotfoundException {
public static void main(String[] args) {
try {
Class.forName("NotfoundException");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
显式加载通常由如下方式:
this.getClass().getClassLoader().getResource("").toString()
NoClassDefFoundError是另一个经常遇到的异常,这个异常在第一次使用命令执行Java类时很可能会碰到,如下面这种情况
java -cp example.jar Examper
在这个jar包里面只有一个类,这个类时net.xx.Exmple ,可能让你感到郁闷的是,命名这个jar包里面有这个类为啥会报这个错呢? 这是因为你在命令行中没有加类的包名,正确的写法是这样的
java cp example.jar net.xx.Example
在JVM的规范中描述了出现NoClassDefFoundError可能的情况就是使用new关键字、属性引用某个类、继承了某个接口或者类,以及方法的某个参数中引用了某个类,这个是出发JVM隐式加载这些类时发现这些类不存在的异常。
解决这个错误的方法就是确保这个类引用的类都在当前的classpath下面
这个异常倒不是很常见,但是出错的话,通常是在JVM启动的时候,如果一不小心将在JVM的某个lib删除了,可能就会报这个错误了,代码如下
package test;
public class NolibException {
public native void nativeMethod();
static {
System.loadLibrary("Nolib");
}
public static void main(String[] args) {
new NolibException().nativeMethod();
}
}
这个错误通常时在解析native标识的方法时JVM找不到对应的本机库文件时出现,代码如下
java.lang.UnsatisfiedLinkError: no Nolib in java.library.path
at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1867)
at java.lang.Runtime.loadLibrary0(Runtime.java:870)
at java.lang.System.loadLibrary(System.java:1122)
at test.NolibException.<clinit>(NolibException.java:6)
Exception in thread "main"
Process finished with exit code 1
这个错误也很常见,通常在程序中出现强制类型转换时出现这个错误,如下面:
package test;
import java.util.HashMap;
import java.util.Map;
public class CastException {
public static Map map = new HashMap(){{
put("a","2");
}};
public static void main(String[] args) {
Integer integer = (Integer) map.get("a");
System.out.println(integer);
}
}
Exception in thread "main" java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Integer
at test.CastException.main(CastException.java:12)
JVM在做类型转换时会按照如下规则进行检查
public static Map<String,Integer> map = new HashMap<String, Integer>(){{
put("a",2);
}};
,这样运行就可以没问题。这个错误在JVM规范中是这样定义的:
将上面的代码例子稍微改一下:
package test;
import java.util.HashMap;
import java.util.Map;
public class CastException {
public static Map<String,Integer> map = new HashMap<String, Integer>(){{
map.put("a",2);
}};
public static void main(String[] args) {
Integer integer = (Integer) map.get("a");
System.out.println(integer);
}
}
这段代码在执行时报错如下:
java.lang.ExceptionInInitializerError
在初始化这个类时,给静态属性map赋值时出现了异常导致抛出错误ExceptionInInitializerError
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有