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

1、java对象初始化过程html

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

    若是该类有父类,则先加载其父类。    测试

    i 初始化该类静态成员spa

    ii 执行该类静态代码块code

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

    若是该类有父类,先建立父类的对象。对象

    i 初始化该类普通成员。  blog

    ii 执行普通代码块。递归

    iii 调用该类构造方法。 get

2、案例测试

  该类对象做为成员变量

public class Info{
    public Info(String s) {
        System.out.println(s);
    }
}

  父类

public class Parent {
    
    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 extends Parent{
    
    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 class InitObjectTest{
    public static void main(String[] args) {
        try{
            //Class.forName("Parent");
            Class.forName("Child");
        }catch(Exception e){
            
        }
        //System.out.println("=============== now , we create an Object below ===========");
        //new Parent();
    }
}

测试结果:

 

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

注意:类的加载只进行一次,以后建立对象将再也不进行类加载,这也是为何静态代码块只执行一次的缘由。

 

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

 

public class InitObjectTest{
    public static void main(String[] args) {
        try{
            //Class.forName("Parent");
            Class.forName("Parent");
        }catch(Exception e){
            
        }
        System.out.println("=============== now , we create an Object below ===========");
        new Parent();
    }
}

 

测试结果:

 

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

 

下面加上子类进行测试。

public class InitObjectTest{
    public static void main(String[] args) {
        try{
            //Class.forName("Parent");
            //Class.forName("Parent");
        }catch(Exception e){
            
        }
        System.out.println("=============== now , we create an Object below ===========");
        new Child();
    }
}

测试结果:

 

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

 

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

public class InitObjectTest{
    public static void main(String[] args) {
        try{
            //Class.forName("Parent");
            Class.forName("Parent");
        }catch(Exception e){
            
        }
        System.out.println("=============== now , we create an Object below ===========");
        new Child();
    }
}

测试结果:

 

首先就加载了父类,在建立子类对象时须要加载子类,加载子类时,须要加载父类,而父类在以前就已经加载过了,因此这里并无再次加载。

 3、总结

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

  本文我的编写,水平有限,若有错误,恳请指出,欢迎讨论分享

相关文章
相关标签/搜索