Class.forName()与xxx.class的区别

所有的类都是在对其第一次使用时,动态加载到JVM。当程序创建第一个对类的静态成员的引用时,就会加载这个类。这也证明了构造器也是类的静态方法,即使在构造器之前并没有static关键字。因此,使用new操作符创建类的对象也会被当作对类的静态成员的引用。

因此Java程序在他开始运行之前并非被完全加载,其各个部分是在需要时才加载的。

类加载器首先槛车这个类的Class对象是否已经加载。如果尚未加载,默认的类加载器就会根据类名称查找.class文件。在这个类的字节码被加载时,它们会接收验证,以确保其没有被破坏。

Class对象尽在需要的时候才被加载。static初始化是在类加载时进行的。

对于Class.forName(),这个方法是Class类的一个static成员。Class对象就和其他对象一样,我们可以获取并操作他的引用。forName()是取得Class对象引用的一种方法。他是用一个包含目标类的文本名的String作为输入参数,返回一个Class对象的引用。forName()方法发现某个累还没被加载,就会主动去加载这个类,在加载过程中,该类的是static子句将会被执行。

Java还提供了另一种方法来生成对Class对象的引用,即使用类字面常量。形如:xxx.class的形式。但是这种形式创建Class对象引用时,不会自动初始化Class对象。初始化被延迟到了对静态方法或者非常数静态域首次引用时才执行。


public class Initable {
	static final int staticFinal = 47;
	static final int staticFianl2 = ClassInitialization.rand.nextInt(1000);
	static {
		System.out.println("Initializing Initable");
	}
}

public class Initable2 {
	static int staticNonFinal = 147;
	static {
		System.out.println("Initializing Initable2");
	}
}

public class Initable3 {
	static int staticNonFinal = 74;
	static {
		System.out.println("Initializing Initable3");
	}
}

public class ClassInitialization {
	public static Random rand = new Random(47);
	public static void main(String[] args) throws ClassNotFoundException {
		Class initable = Initable.class;
		System.out.println("after creating Initable ref");
		//Does not trigger initialization
		System.out.println(Initable.staticFinal);
		//Does trigger initialization
		System.out.println(Initable.staticFianl2);
		//Does trigger initialization
		System.out.println(Initable2.staticNonFinal);
		Class initable3 = Class.forName("chapter13.t2.Initable3");
		System.out.println("after creating Initable3 ref");
		System.out.println(Initable3.staticNonFinal);
	}
}


输出:

after creating Initable ref
47
Initializing Initable
258
Initializing Initable2
147
Initializing Initable3
after creating Initable3 ref
74


从对initable引用的创建中可以看到,仅使用.class语法来获取对类的引用不会发生初始化。但是Class.forName()立即就进行了初始化。如果一个static final值是“编译期常量”,就像Initable.staticFinal那样,那么这个值不需要对Initable类进行初始化就可以被读取。但是如果只是将一个域设置为static和final的,还不足以确保这种行为,例如Initable.staticFianl2的访问将强制进行类型的初始化,因为他不是一个编译器常量。



  • 4
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
java.lang.NoClassDefFoundError: Could not initialize class异常通常是由于类的静态初始化块或静态变量的初始化过程中发生了异常导致的。这个异常表示在类加载过程中,虚拟机无法找到或初始化指定的类。 对于引用中的异常"java.lang.NoClassDefFoundError: Could not initialize class xxx",其中的"xxx"是指具体的类名。这个异常的原因可能是以下几种情况之一: 1. 缺少相关的依赖库或jar包。 2. 类的静态初始化块或静态变量的初始化过程中发生了异常。 3. 类的初始化过程中发生了其他异常,例如访问权限问题或反射异常等。 解决这个异常的方法可以尝试以下几种方式: 1. 检查相关的依赖库或jar包是否存在,并确保它们的版本和路径正确。 2. 检查类的静态初始化块或静态变量的初始化过程中是否有异常抛出,并修复这些异常。 3. 检查类的初始化过程中是否有其他异常抛出,并修复这些异常。 4. 检查类的访问权限是否正确,并确保没有其他限制导致类无法被初始化。 5. 如果使用了反射相关的代码,检查反射相关的异常,并修复这些异常。 以下是一个示例代码,演示了如何处理java.lang.NoClassDefFoundError异常: ```java try { // 尝试初始化类 Class.forName("org.mockito.Mockito"); } catch (ClassNotFoundException e) { // 类未找到异常 e.printStackTrace(); } catch (ExceptionInInitializerError e) { // 类的静态初始化过程中发生了异常 e.printStackTrace(); } catch (Throwable t) { // 其他异常 t.printStackTrace(); } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值