Java初始化顺序

 JVM 类加载过程 :

   类从加载到虚拟机到卸载,它的整个生命周期包括:加载(Loading),验证(Validation),准备(Preparation),解析(Resolution),初始化(Initialization),使用(Using)和卸载(Unloading)。其中,验证、准备和解析部分被称为连接(Linking)。

  blob.png

  加载:

    在加载阶段,虚拟机主要完成三件事:     

        1.通过一个类的全限定名来获取定义此类的二进制字节流。

        2.将这个字节流所代表的静态存储结构转化为方法区域的运行时数据结构。

        3.在Java堆中生成一个代表这个类的java.lang.Class对象,作为方法区域数据的访问入口。

  

  验证:

   验证阶段作用是保证Class文件的字节流包含的信息符合JVM规范,不会给JVM造成危害。如果验证失败,就会抛出一个java.lang.VerifyError异常或其子类异常。验证过程分为四个阶段:   

    1.文件格式验证:验证字节流文件是否符合Class文件格式的规范,并且能被当前虚拟机正确的处理。

    2.元数据验证:是对字节码描述的信息进行语义分析,以保证其描述的信息符合Java语言的规范。

    3.字节码验证:主要是进行数据流和控制流的分析,保证被校验类的方法在运行时不会危害虚拟机。

    4.符号引用验证:符号引用验证发生在虚拟机将符号引用转化为直接引用的时候,这个转化动作将在解析阶段中发生。

  

  准备:

   准备阶段为变量分配内存并设置类变量的初始化。在这个阶段分配的仅为类的变量(static修饰的变量),而不包括类的实例变量。对已非final的变量,JVM会将其设置成“零值”,而不是其赋值语句的值:

   pirvate static int size = 12;

   那么在这个阶段,size的值为0,而不是12。 final修饰的类变量将会赋值成真实的值。

 

  解析:

   解析过程是将常量池内的符号引用替换成直接引用。主要包括四种类型引用的解析。类或接口的解析、字段解析、方法解析、接口方法解析。

 

  初始化:

   在准备阶段,类变量已经经过一次初始化了,在这个阶段,则是根据程序员通过程序制定的计划去初始化类的变量和其他资源。这些资源有static{}块,构造函数,父类的初始化等。

    至于使用和卸载阶段阶段,这里不再过多说明,使用过程就是根据程序定义的行为执行,卸载由GC完成。


 Java初始化顺序过程:

   1  无继承情况下的Java初始化顺序: 

        class Sample
    {
          Sample(String s)
          {
                System.out.println(s);
          }
          Sample()
          {
                System.out.println("Sample默认构造函数被调用");
          }
    }
    class Test{
          static Sample sam=new Sample("静态成员sam初始化");
          Sample sam1=new Sample("sam1成员初始化");
          static{
                System.out.println("static块执行");
                if(sam==null)System.out.println("sam is null");
                sam=new Sample("静态块内初始化sam成员变量");
                }
          Test()
          {
                System.out.println("Test默认构造函数被调用");
          }
    
    }
    //主函数
     public static void  main(String  str[])
    {
    
         Test a=new Test();
    
     }

    输出结果为:

        静态成员sam初始化     —–静态成员初始化

        static块执行          —–静态块被执行

        静态块内初始化sam成员变量 —-静态块执行

        sam1成员初始化      —–普通成员初始化

        Test默认构造函数被调用  –—-构造函数执行

    

   由此可以得出结论:

    静态成员变量首先初始化(注意,Static可以看做一个静态成员,其执行顺序和其在类中申明的顺序有关)

    b 普通成员初始化

        c 构造代码块初始化。

        执行构造函数.

  

 

     对于静态成员(static块可以看成普通的一个静态成员,其并不一定在类初始化时首先执行)和普通成员,其初始化顺序只与其在类定义中的顺序有关,和其他因素无关。

例如下面的例子:

   

        class Test{
          static{
                System.out.println("static 块 1  执行");
                }
          static Sample staticSam1=new Sample("静态成员staticSam1初始化");
          Sample sam1=new Sample("sam1成员初始化");
          static Sample staticSam2=new Sample("静态成员staticSam2初始化");
          static{
                System.out.println("static 块 2  执行");
                }
          Test()
          {
                System.out.println("Test默认构造函数被调用");
          }
          Sample sam2=new Sample("sam2成员初始化");
      }

  则结果为:

    static 块 1  执行

    静态成员staticSam1初始化

    静态成员staticSam2初始化

    static 块 2  执行 

                          ——–静态成员

    sam1成员初始化

    sam2成员初始化

                          ——–普通成员

    Test默认构造函数被调用

                          ——–构造函数

    

2 Java继承情况下的初始化顺序:

class Test{
      static{
            System.out.println("父类static 块 1  执行");
            }
      static Sample staticSam1=new Sample("父类 静态成员staticSam1初始化");
      Sample sam1=new Sample("父类 sam1成员初始化");
      static Sample staticSam2=new Sample("父类 静态成员staticSam2初始化");
      static{
            System.out.println("父类 static 块 2  执行");
            }
      Test()
      {
            System.out.println("父类 Test默认构造函数被调用");
      }
      Sample sam2=new Sample("父类 sam2成员初始化");
}
class TestSub extends Test
{
      static Sample staticSamSub=new Sample("子类 静态成员staticSamSub初始化");
      TestSub()
      {
            System.out.println("子类 TestSub 默认构造函数被调用");
      }
      Sample sam1=new Sample("子类 sam1成员初始化");
      static Sample staticSamSub1=new Sample("子类 静态成员staticSamSub1初始化");
      
      static{System.out.println("子类 static 块  执行");}
      Sample sam2=new Sample("子类 sam2成员初始化");
}

 

    执行结果:

        父类 static 块 1  执行

        父类 静态成员staticSam1初始化

        父类 静态成员staticSam2初始化

        父类 static 块 2  执行

                                ——–父类静态成员初始化

        子类 静态成员staticSamSub初始化

        子类 静态成员staticSamSub1初始化

        子类 static 块  执行

                                ——-子类静态成员初始化

        父类 sam1成员初始化

        父类 sam2成员初始化

          父类 构造代码块

        父类 Test默认构造函数被调用       

                                ——-父类普通成员初始化和构造函数执行

        子类 sam1成员初始化

        子类 sam2成员初始化

           子类 构造代码块

        子类 TestSub 默认构造函数被调用

                                ——-父类普通成员初始化和构造函数执行

    

    由此得出Java初始化顺序结论:

    1 继承体系的所有静态成员初始化(先父类,后子类)

    2 父类初始化完成(普通成员的初始化–>构造代码块 —>构造函数的调用)

    3 子类初始化(普通成员–>构造代码块 —>构造函数)

    blob.png

     注意:

      调用类里面的final static变量是不会初始化任何类及静态代码块信息。

      调用父类的static变量或者是static方法只会初始化父类的静态代码块信息。

      调用子类的static变量或者是static方法 同时先初始化父类的静态代码信息。



   Person p = new Person("zhangsan",20);JVM运行到这句话时,做了什么事情?

        1,因为new用到了Person.class.所以会先找到Person.class文件并加载到内存中。

        2,执行该类中的static代码块,如果有的话,给Person.class类进行初始化。

        3,在堆内存中开辟空间,分配内存地址。

        4,在堆内存中建立对象的特有属性。并进行默认初始化。

        5,对对象属性进行显示初始化。

        6,对对象进行构造代码块初始化。

        7,对对象进行对应的构造函数初始化。

        8,将内存地址赋给栈内存中的p变量。


发表评论