大约阅读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中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虚拟就会对其初始化,如下六种情况为主动使用:
- 当创建某个类的新实例时(如通过new或者反射,克隆,反序列化等)
- 当调用某个类的静态方法时
- 当使用某个类或接口的静态字段时
- 当调用Java API中的某些反射方法时,比如类Class中的方法,或者java.lang.reflect中的类的方法时
- 当初始化某个子类时
- 当虚拟机启动某个被标明为启动类的类(即包含main方法的那个类)
Java编译器会收集所有的类变量初始化语句和类型的静态初始化器,将这些放到一个特殊的方法中:clinit 。
重点:使用final修饰的静态字段,在调用的时候不会对类进行初始化!
BAT面试题