类加载的三个准备工作

当使用“.class”来创建对Class对像的引用时,不会自动地初始化该Class对像。为了使用类而做的准备工作实际包含三个步骤:

1、加载,这是由类加载器执行的。该步骤将查找字节码(通常在classpath所指定的路径中查找,但这并非是必需的),并从这些字节码中创建一个Class对像。

2、链接。在链接阶段将验证类中的字节码,为静态城分配存储空间,并且如果必需的话,将解析这个类创建的对其它类的所有引用。

3、初始化。如果该类具有超类,则对其初始化,执行静态初始化器和静态初始化块。

初始化被延迟到了对静态方法(构造器隐式的是静态的)或者非常数静态域进行首次引用时才执行

例:

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

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

public class Initbale3 {
	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=ShapeDemo3.class;
		System.out.println("AFTER");
		System.out.println(ShapeDemo4.staticFinal);
		System.out.println(ShapeDemo4.staticFinal2);
		System.out.println(Initbale2.staticNonFinal);
		Class initaClass=Class.forName("com.shape.Initbale3");
		System.out.println("After");
		System.out.println(Initbale3.staticNonFinal);
	}
}


result::

AFTER
47
Initializing Initable
258
Initializing Initable2
147
Initializing Initable3
After
74


从上面程序运行的结果可以看出初始化尽可能的实现“惰性”。从对SheapDemo4引用的创建中可以看到,当只使用.class语法来获得对类的引用时不会引发初始化。但是,为了产生Class引用,Class.forName()立即就进行了初始化。

如果一个static final值是“编译期常量”,那么这个值不需要对类进行初始化就可以被读取,如果例子中的“static final int staticFinal = 47;”但是如果只是将一个域设置为static或final时,还不足以确保这种行为,例如对“static int staticNonFinal=147;”进行访问时将强制进行初始化,因为它不是一个编译期常量。

如果一个static域不是final的,那么在对它访问时,总是要求在它被读取之前,要先进行链接(即为这个域分配存储空间)和初始化(初始化该在存储空间),例如对Initable.staticNonFinal的访问中所看到的结果。

第一次博客,基本按书上意思转达。因自己写作水平有限,请朋友们多多指教,以后会抽更多的时间记录。奋斗

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值