NoClassDefFoundError异常的解决办法

NoClassDefFoundError产生的原因有好几种,这里记录静态变量或静态块引起的。具体抛出的异常类似:
java.lang.NoClassDefFoundError: Could not initialize class xxx

JVM在加载类的时候,会初始化类里的静态变量,或执行静态块,如果这个时候抛出了异常,该类就会加载失败,那么以后任何使用到这个类的地方,都会抛出NoClassDefFoundError异常,如下面的例子:

public class TestNoClassDefFoundError {
    public static void main(String[] args) throws InterruptedException {
        TestNoClassDefFoundError sample = new TestNoClassDefFoundError();
        sample.getClassWithInitErrors();
    }

    private void getClassWithInitErrors() throws InterruptedException {
        System.out.println("第一次new");
        Thread.sleep(500);
        try {
            //第一次new ClassWithInitErrors类,JVM会加载该类,初始化该类的静态变量或执行静态块
            new ClassWithInitErrors();
        } catch (Throwable t) {
            //因为初始化静态变量失败,所以加载类失败。
            t.printStackTrace();
        }

        Thread.sleep(500);
        System.out.println("-----------------------------------------------------");
        System.out.println("第二次new");
        Thread.sleep(500);
        try {
            //第二次new ClassWithInitErrors类,JVM不会再加载该类,而是抛出NoClassDefFoundError异常
            new ClassWithInitErrors();
        } catch (Throwable t) {
            t.printStackTrace();
        }
        Thread.sleep(500);
        System.out.println("-----------------------------------------------------");
        System.out.println("第三次new");
        Thread.sleep(500);
        try {
            //第三次new ClassWithInitErrors类,JVM不会再加载该类,而是抛出NoClassDefFoundError异常
            new ClassWithInitErrors();
        } catch (Throwable t) {
            t.printStackTrace();
        }
    }
}

class ClassWithInitErrors {
    static int data = 1 / 0;
}

执行结果如下:

第一次new
java.lang.ExceptionInInitializerError
    at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:12)
    at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)
Caused by: java.lang.ArithmeticException: / by zero
    at ClassWithInitErrors.<clinit>(TestNoClassDefFoundError.java:42)
    ... 2 more
-----------------------------------------------------
第二次new
java.lang.NoClassDefFoundError: Could not initialize class ClassWithInitErrors
    at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:24)
    at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)
-----------------------------------------------------
第三次new
java.lang.NoClassDefFoundError: Could not initialize class ClassWithInitErrors
    at TestNoClassDefFoundError.getClassWithInitErrors(TestNoClassDefFoundError.java:34)
    at TestNoClassDefFoundError.main(TestNoClassDefFoundError.java:4)

上述代码,第二次及以后使用到ClassWithInitErrors类的地方,都只会抛出NoClassDefFoundError错误,而且通常我们查看日志的时候,都会先看最新的日志,所以就不容易找到加载类失败的真正原因,对于这种情况,我们应该搜索java.lang.ExceptionInInitializerError错误,找到第一次加载类的时候的日志,才能看到失败的原因。

实例
错误日志:java.lang.NoClassDefFoundError: Could not initialize class sun.awt.X11Graphi

原因分析:启动tomcat的用户无法访问Xserver,包括Xserver没有启动或者没有权限访问。
对于一个Java服务器来说经常要处理一些图形元素,例如地图的创建或者图形和图表等。这些API基本上总是需要运行一个X-server以便能使用AWT(Abstract Window Toolkit,抽象窗口工具集)。
解决办法:vi /usr/local/tomcat/bin/catalina.sh 中增加 -Djava.awt.headless=true
如下:
JAVA_OPTS="$JAVA_OPTS -Djava.awt.headless=true"
&
JAVA_OPTS='-server -Xms2048m -Xmx2048m -XX:PermSize=256m -XX:MaxPermSize=512m -Djava.awt.headless=true'
添加完之后进行tomcat重启即可。
参考文章:

给TA买糖
共{{data.count}}人
人已赞赏
系统运维

Linux运维基础技能: 脚本编程与Linux命令

2020-5-10 17:10:41

系统运维

LINUX中的定时任务设置crontab

2020-5-12 10:33:06

⚠️
Golurending上的部份代码及教程来源于互联网,仅供网友学习交流,未经Golurending作者或上传书面授权,请勿作他用。
若您的权利被侵害,请联系站长 邮箱codesns#163.com(#-@) 或 点击右侧 私信:管理员 反馈,我们将尽快处理。
0 条回复 A文章作者 M管理员
    暂无讨论,说说你的看法吧
个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索