Java初始化顺序

来自:Java编程思想


了解包括继承在内的初始化全过程,以对所发生的一切有个全局的把握,是很有益的。

请看下例:

package org.easyliu.demo2.inittest;

class Insect {

	private int i = 9;
	protected int j;

	Insect() {
		System.out.println("i = " + i + ", j = " + j);
		j = 39;
	}

	private int p = print("Insect.p initialized");
	private static int x1 = print("static Insect.x1 initialized");
	static { 

		System.out.println("static block Insect");
	}

	static int print(String s) {
		System.out.println(s);
		return 47;
	}
}

public class Beetle extends Insect {

	private int k = print("Beetle.k initialized");

	public Beetle() {
		System.out.println("k = " + k);
		System.out.println("j = " + j);
	}

	private static void dd() {
		System.out.println("dd ");
	}

	private static int x2 = print("static Beetle.x2 initialized");
	
	static {
		System.out.println("static block Beetle");
	}
	public static void main(String[] args) {
		System.out.println("Beetle constructor");
		Beetle b = new Beetle();

	}

}



输出结果如下:

static Insect.x1 initialized
static block Insect
static Beetle.x2 initialized
static block Beetle
Beetle constructor
Insect.p initialized
i = 9, j = 0
Beetle.k initialized
k = 47
j = 39


分析:

首先当我们运行此程序的时候,主线程启动,所发生的第一件事情就是试图访问Beetle.main()(一个static方法),于是加载器开始查找Beetle.class这个类,找到这个类后,开始加载这个类,在加载的过程发现有extends关键字,所以加载器开始去加载insect.class这个类,(如果在加载这个类的时候如果发现这个类还有基类,那么insect的直接基类就会被加载,如果还有基类,以此类推)此时因为insect没有基类了,所以不需要再加载,此时(加载insect类时)insect类的静态代码将执行(静态成员的初始化x1),所以此时打印出: static Insect.x1 initialized, 然后是static块初始化(Static可以看做一个静态成员,其执行顺序和其在类中申明的顺序有关),所以打印出:static block Insect。当insect的静态代码执行完后, Beetle的静态代码开始执行 (初始化x2), 所以打印出:static Beetle.x2 initialized,然后是static块初始化,所以打印出:static block Beetle;

到现在为止,所有基类以及静态的初始化动作都已经执行完了,可以创建对象了。

这时候执行System.out.println("Beetle constructor");所以打印出:Beetle constructor;


下面开始执行Beetle b = new Beetle();此时首先为Beetle分配内存空间,并且都初始化为0,注意boolean类型变量初始化为false;

在Beetle内的代码执行前,首先会调用基类的构造器,在基类(insect)的构造器内的代码执行前,首先要执行基类的非静态成员的初始化,此时p被初始化为47,并且打印出:Insect.p initialized;当执行初始化后,基类的构造器里的代码System.out.println("i = " + i + ", j = " + j)开始执行,所以打印出:i = 9, j = 0;此时j被初始化为39;


当基类的构造器的代码执行完后,首先要对Beetle类的非静态成员进行显示的初始化,此时K被初始化为47,并且打印出:Beetle.k initialized;等初始化完成后,Beetle构造器里的代码开始执行,所以打印出:K=47;J=39,至此分析完毕!


值得注意的是:静态成员初始化是从最顶层的类开始的,因为导出类的static初始化可能会依赖于基类成员能否被正确初始化。


总结一下初始化顺序:

类的静态成员(static子块也可以看做静态成员,静态成员的初始化顺序由定义的顺序决定)初始化(有父类的先父类,然后自己)->

然后main方法执行->

类初始化{非静态成员的初始化->构造函数(有父类的先父类的非静态成员初始化,然后构造函数)}


下面是框图:




  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值