Java類初始化執行流程


    測試代碼:  

package com.test.ClassLaoderTest;

public  class test1 {
    public static String s_variable = "靜態變量";
    public String init_variable = "公開的變量";
    private String p_variable = "私有的變量";
    //靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("靜態代碼塊初始化執行了");
    }

    //初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("初始化代碼塊執行淪");
    }

    //構造方法
    public test1(){
        System.out.println("我是構造方法");
    }

    public static void main(String[] args) {

    }

}

  直接運行:

    

 

 

  main方法里面不做任何調用的情況下,自動調用的是靜態代碼塊和靜態變量

  (2)調用靜態變量和靜態方法:

    測試代碼:    

package com.test.ClassLaoderTest;


public class test1 {
    public static String s_variable = "靜態變量";
    public String init_variable = "公開的變量";
    private String p_variable = "私有的變量";
    //靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("靜態代碼塊初始化執行了");
    }

    //初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("初始化代碼塊執行淪");
    }

    //構造方法
    public test1(){
        System.out.println("我是構造方法");
    }

    //靜態方法
    public static void test1(){
        System.out.println("這是靜態方法");
    }
    public static void main(String[] args) {
        System.out.println(test1.s_variable);
        test1.test1();
    }

}

  運行:

  

 

 

  結論:當我調用靜態方法/靜態變量時,只會家在靜態代碼塊,其余的代碼塊/構造方法不會被加載

    

  (3)創建對象:

     

package com.test.ClassLaoderTest;


public class test1 {
    public static String s_variable = "靜態變量";
    public String init_variable = "公開的變量";
    private String p_variable = "私有的變量";
    //靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("靜態代碼塊初始化執行了");
    }

    //初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("初始化代碼塊執行了");
    }

    //構造方法
    public test1(){
        System.out.println("我是構造方法");
    }

    //靜態方法
    public static void test1(){
        System.out.println("這是靜態方法");
    }
    public static void main(String[] args) {
        test1 t1 = new test1();
    }

}

  運行結果:

    

 

  輸出內容: 

    

靜態變量
靜態代碼塊初始化執行了
公開的變量
私有的變量
初始化代碼塊執行了
我是構造方法

  結論:當創建對象/實例化的時候,調用順序:靜態代碼塊->初始化代碼->構造方法,最后執行的才是構造方法

  

  (4)有繼承關系下的類初始化執行流程:

   環境:

    父類:

     

package com.test.ClassLaoderTest;

public class father {
    public static String s_variable = "父類靜態變量";
    public String init_variable = "父類公開的變量";
    private String p_variable = "父類私有的變量";
    //父類靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("父類靜態代碼塊初始化執行了");
    }

    //父類初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("父類初始化代碼塊執行了");
    }

    //構造方法
    public father(){
        System.out.println("我是父類構造方法");
    }

    //父類靜態方法
    public static void test1(){
        System.out.println("這是父類靜態方法");
    }
}

  

    test1.java:

    繼承其父類father:

package com.test.ClassLaoderTest;

public class test1 extends father{
    public static String s_variable = "子類靜態變量";
    public String init_variable = "子類公開的變量";
    private String p_variable = "子類私有的變量";
    //子類靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("子類靜態代碼塊初始化執行了");
    }

    //子類初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("子類初始化代碼塊執行了");
    }

    //子類構造方法
    public test1(){
        System.out.println("我是子類構造方法");
    }

    //子類靜態方法
    public static void test1(){
        System.out.println("這是子類靜態方法");
    }

    public static void main(String[] args) {
        
    }
}

  

  main方法不做任何操作,運行:

    

 

 只要extends繼承了,優先調用父類靜態代碼塊

 

(5)有繼承關系下的調用靜態方法:

    修改子類即可:

package com.test.ClassLaoderTest;

public class test1 extends father{
    public static String s_variable = "子類靜態變量";
    public String init_variable = "子類公開的變量";
    private String p_variable = "子類私有的變量";
    //子類靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("子類靜態代碼塊初始化執行了");
    }

    //子類初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("子類初始化代碼塊執行了");
    }

    //子類構造方法
    public test1(){
        System.out.println("我是子類構造方法");
    }

    //子類靜態方法
    public static void test1(){
        System.out.println("這是子類靜態方法");
    }

    public static void main(String[] args) {
        test1.test1();
        father.test1();
    }
}

 

 

   運行:

      

 

 

 

  結果:

父類靜態變量
父類靜態代碼塊初始化執行了
子類靜態變量
子類靜態代碼塊初始化執行了
這是子類靜態方法
這是父類靜態方法

  main方法中,誰優先調用靜態方法,就優先加載誰

 

 (6)有繼承關系下的創建對象:

    代碼:

  

package com.test.ClassLaoderTest;

public class test1 extends father{
    public static String s_variable = "子類靜態變量";
    public String init_variable = "子類公開的變量";
    private String p_variable = "子類私有的變量";
    //子類靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("子類靜態代碼塊初始化執行了");
    }

    //子類初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("子類初始化代碼塊執行了");
    }

    //子類構造方法
    public test1(){
        System.out.println("我是子類構造方法");
    }

    //子類靜態方法
    public static void test1(){
        System.out.println("這是子類靜態方法");
    }

    public static void main(String[] args) {
        test1 t1 =new test1();
    }
}

  運行:

   

 

 

   結果:

父類靜態變量
父類靜態代碼塊初始化執行了
子類靜態變量
子類靜態代碼塊初始化執行了
父類公開的變量
父類私有的變量
父類初始化代碼塊執行了
我是父類構造方法
子類公開的變量
子類私有的變量
子類初始化代碼塊執行了
我是子類構造方法

  結論:通過結果會發現,不管是子類還是父類靜態代碼塊,靜態代碼塊在哪里都是爸爸級別,最先加載的,當創建test1對象的時候,優先加載的是父類代碼塊,那么他的初始化執行流程如下:父類靜態代碼塊>子類靜態代碼塊>父類初始化代碼塊>父類構造方法>子類代碼塊>子類構造方法

   

   

 (7)  有繼承關系下的創建父類對象:

   

package com.test.ClassLaoderTest;

public class test1 extends father{
    public static String s_variable = "子類靜態變量";
    public String init_variable = "子類公開的變量";
    private String p_variable = "子類私有的變量";
    //子類靜態代碼塊
    static {
        System.out.println(s_variable);
        System.out.println("子類靜態代碼塊初始化執行了");
    }

    //子類初始化代碼塊
    {
        System.out.println(init_variable);
        System.out.println(p_variable);
        System.out.println("子類初始化代碼塊執行了");
    }

    //子類構造方法
    public test1(){
        System.out.println("我是子類構造方法");
    }

    //子類靜態方法
    public static void test1(){
        System.out.println("這是子類靜態方法");
    }

    public static void main(String[] args) {
        father father = new father();
    }
}

 

 運行:

  

 

 

  結果:

    

父類靜態變量
父類靜態代碼塊初始化執行了
子類靜態變量
子類靜態代碼塊初始化執行了
父類公開的變量
父類私有的變量
父類初始化代碼塊執行了
我是父類構造方法

  結論:優先執行的是兩個類的靜態代碼塊,然后是父類型的代碼塊和構造方法,而子類的代碼塊和構造方法沒有被執行是因為沒有實例化子類,所以肯定是沒有他的,那么只有在創建對象的時候,才會調用代碼塊和構造方法


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM