​大约阅读2分钟,写于 2021 0408 22:31 北京石景山家里台灯下​


java.lang.NoClassDefFoundError:无法初始化类XXX



文章目录


1. 场景

在本地,debug模式启动调试,代码大概是如下:

public class PropHolder {
public static Properties prop;

static {
//code for loading properties from file
}
}

// Referencing the class somewhere else:
Properties prop = PropHolder.prop;

2. 问题


java.lang.NoClassDefFoundError:无法初始化类XXX


在运行一段时间后又恢复了正常(真的很奇怪,文章末尾有参考案例)

3. 原因

就现在而言,我有一个静态初始化的变量,该变量依赖于属性文件中的缺失条目。

当然,不排除有其他问题,了解底层 JVM 类加载过程,对解决问题很有帮助。(搜 JavaPub版,掌握底层逻辑)

4. 解决思路

意味着 配置文件 中的属性不正确或为空!



补充缺失参数



如果配置文件没问题,参考这篇文章:http://www.voidcn.com/article/p-bayljcop-bmr.html

大概思路就是说初始化速度不一致,导致初始化失败,改成单例,交由Spring管理,依赖的属性交由Spring注入。
java.lang.NoClassDefFoundError:无法初始化类XXX_spring



如果没有解决你的问题,请继续看

Java中static块执行时机

public class StaticDemo {

public static final String INIT = "init";
static {
System.out.println("------StaticDemo----");
}
}

public class TestStatic {

public static void main(String[] args) {
System.out.println(StaticDemo.INIT);
}
}
//打印
init

怎么没有执行 static 块的代码呢?

最后发现是因为调用的静态属性被final修饰导致的。去掉final修饰,结果OK!

public class StaticDemo {

//去掉 final
public static String INIT = "init";
static {
System.out.println("------StaticDemo----");
}
}

public class TestStatic {

public static void main(String[] args) {
System.out.println(StaticDemo.INIT);
}
}
//打印结果
------StaticDemo----
init

这里就和类加载机制中的 初始化 相关! ​​类被加载了不一定就会执行静态代码块,只有一个类被主动使用的时候,静态代码才会被执行!​


当一个类被主动使用时,Java虚拟就会对其初始化,如下六种情况为主动使用:


  1. 当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)
  2. 当调用某个类的静态方法时
  3. 当使用某个类或接口的静态字段时
  4. 当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时
  5. 当初始化某个子类时
  6. 当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)

Java编译器会收集所有的类变量初始化语句和类型的静态初始化器,将这些放到一个特殊的方法中:clinit 。

重点:使用final修饰的静态字段,在调用的时候不会对类进行初始化!

​BAT面试题​