java初始化代码块是_java 对象的初始化流程(静态成员、静态代码块、普通代码块、构造方法)...

一、java对象初始化过程

第一步,加载该类,一个java对象在初始化前会进行类加载,在JVM中生成Class对象。加载一个类会进行如下操作,下面给出递归描述。(关于Class对象详见反射

如果该类有父类,则先加载其父类。

i 初始化该类静态成员

ii 执行该类静态代码块

第二步,创建对象,如果该类有父类,则创建对象时会先创建其父类的对象,外层包裹子类的属性和方法,然后返回子类的引用,下面给出递归描述。

如果该类有父类,先创建父类的对象。

i 初始化该类普通成员。

ii 执行普通代码块。

iii 调用该类构造方法。

二、案例测试

该类对象作为成员变量

public classInfo{publicInfo(String s) {

System.out.println(s);

}

}

父类

public classParent {public static Info info = new Info("Parent static member"); //静态成员

public Info info2 = new Info("Parent common member"); //普通成员

static { //静态代码块

System.out.println("parent static block");

}

{//普通代码块

System.out.println("parent common block");

}public Parent() { //父类构造方法

System.out.println("Parent.Parent()");

}

}

子类

public class Child extendsParent{public static Info info = new Info("Child static member"); //静态成员

public Info info2 = new Info("Child common member"); //普通成员

static { //静态代码块

System.out.println("Child static block");

}

{//普通代码块

System.out.println("Child common block");

}public Child() { //子类构造方法

System.out.println("Child.Child()");

}

}

下面测试类的加载过程,我们不创建对象,而是直接加载类,并且是加载子类

public classInitObjectTest{public static voidmain(String[] args) {try{//Class.forName("Parent");

Class.forName("Child");

}catch(Exception e){

}//System.out.println("=============== now , we create an Object below ===========");//new Parent();

}

}

测试结果:

e50a6ed7c03ed2650d7bb2cc7f6b56be.png

测试结果符合上面所写的加载类的规则,先初始化父类静态成员,再执行父类静态块,然后初始化子类静态成员,最后执行子类静态块。我们可以看到静态成员确实在类加载时初始化。

注意:类的加载只进行一次,之后创建对象将不再进行类加载,这也是为什么静态代码块只执行一次的原因。

下面,将父类加载与创建父类对象分开,观察测试结果

public classInitObjectTest{public static voidmain(String[] args) {try{//Class.forName("Parent");

Class.forName("Parent");

}catch(Exception e){

}

System.out.println("=============== now , we create an Object below ===========");newParent();

}

}

测试结果:

002cc150260ec8bb30db0743692bc0c5.png

测试结果符合上面的规则,我们先显示的加载了Parent类,所以后面在new Parent()时就没有再加载类了。在创建对象时,先初始化普通成员,再执行普通代码块,最后调用构造方法。

下面加上子类进行测试。

public classInitObjectTest{public static voidmain(String[] args) {try{//Class.forName("Parent");//Class.forName("Parent");

}catch(Exception e){

}

System.out.println("=============== now , we create an Object below ===========");newChild();

}

}

测试结果:

4da50041ae4945e1529262600c99a4fc.png

当我们没有显示的加载类时,new对象时,会自动加载类。而输出的前四行就是,加载类的反应。后面的六行是创建对象的反应,先初始父类的普通成员,再执行父类的普通代码块,然后调用父类构造方法,然后进行子类的类似操作。完全符合上面描述的创建过程。

下面测试,先加载父类,然后直接创建子类对象。

public classInitObjectTest{public static voidmain(String[] args) {try{//Class.forName("Parent");

Class.forName("Parent");

}catch(Exception e){

}

System.out.println("=============== now , we create an Object below ===========");newChild();

}

}

测试结果:

e3dfbe34a88976b5105d70c45df86aaa.png

首先就加载了父类,在创建子类对象时需要加载子类,加载子类时,需要加载父类,而父类在之前就已经加载过了,所以这里并没有再次加载。

三、总结

到此,静态成员、静态代码块、普通成员、普通代码块、构造方法以及父类的这些模块之间的执行时序就讲完了。分成加载和创建两个步骤来看,十分清晰,每个步骤中又涉及父类的加载,这是一个递归的过程。成员的初始化在代码块的执行之前,因为代码块可能会操作成员。代码块常常用于初始化成员。

本文个人编写,水平有限,如有错误,恳请指出,欢迎讨论分享

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值