1. 程式人生 > >Java中init和clinit區別完全解析

Java中init和clinit區別完全解析

init和clinit區別

①init和clinit方法執行時機不同

init是物件構造器方法,也就是說在程式執行 new 一個物件呼叫該物件類的 constructor 方法時才會執行init方法,而clinit是類構造器方法,也就是在jvm進行類載入—–驗證—-解析—–初始化,中的初始化階段jvm會呼叫clinit方法。

②init和clinit方法執行目的不同

init is the (or one of the) constructor(s) for the instance, and non-static field initialization. 
clinit are the static initialization blocks for the class, and static field initialization. 
上面這兩句是Stack Overflow上的解析,很清楚init是instance例項構造器,對非靜態變數解析初始化

,而clinit是class類構造器對靜態變數,靜態程式碼塊進行初始化。看看下面的這段程式就很清楚了。

class X {

   static Log log = LogFactory.getLog(); // <clinit>

   private int x = 1;   // <init>

   X(){
      // <init>
   }

   static {
      // <clinit>
   }

}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

clinit詳解

在準備階段,變數已經賦過一次系統要求的初始值,而在初始化階段,則根據程式設計師通過程式制定的主觀計劃去初始化類變數和其他資源,或者可以從另外一個角度來表達:初始化階段是執行類構造器<clinit>()方法的過程。

①<clinit>()方法是由編譯器自動收集類中的所有類變數的賦值動作和靜態語句塊(static{}塊)中的語句合併產生的,編譯器收集的順序是由語句在原始檔中出現的順序所決定的,靜態語句塊中只能訪問到定義在靜態語句塊之前的變數,定義在它之後的變數,在前面的靜態語句塊可以賦值,但是不能訪問如下程式碼

public class Test{
static{
i=0;//給變數賦值可以正常編譯通過
System.out.print(i);//這句編譯器會提示"非法向前引用"
}
static int i=1;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

虛擬機器會保證在子類的<clinit>()方法執行之前,父類的<clinit>()方法已經執行完畢。 因此在虛擬機器中第一個被執行的<clinit>()方法的類肯定是java.lang.Object。由於父類的<clinit>()方法先執行,也就意味著父類中定義的靜態語句塊要優先於子類的變數賦值操作,如下程式碼中,欄位B的值將會是2而不是1。

static class Parent{
    public static int A=1;
    static{
    A=2;}
    static class Sub extends Parent{
    public static int B=A;
    }
    public static void main(String[]args){
    System.out.println(Sub.B);
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

介面中不能使用靜態語句塊,但仍然有變數初始化的賦值操作,因此介面與類一樣都會生成<clinit>()方法。 但介面與類不同的是,執行介面的<clinit>()方法不需要先執行父介面的<clinit>()方法。 只有當父介面中定義的變數使用時,父接口才會初始化。 另外,介面的實現類在初始化時也一樣不會執行介面的<clinit>()方法。 
注意:介面中的屬性都是static final型別的常量,因此在準備階段就已經初始化話。

init和clinit區別

①init和clinit方法執行時機不同

init是物件構造器方法,也就是說在程式執行 new 一個物件呼叫該物件類的 constructor 方法時才會執行init方法,而clinit是類構造器方法,也就是在jvm進行類載入—–驗證—-解析—–初始化,中的初始化階段jvm會呼叫clinit方法。

②init和clinit方法執行目的不同

init is the (or one of the) constructor(s) for the instance, and non-static field initialization. 
clinit are the static initialization blocks for the class, and static field initialization. 
上面這兩句是Stack Overflow上的解析,很清楚init是instance例項構造器,對非靜態變數解析初始化,而clinit是class類構造器對靜態變數,靜態程式碼塊進行初始化。看看下面的這段程式就很清楚了。

class X {

   static Log log = LogFactory.getLog(); // <clinit>

   private int x = 1;   // <init>

   X(){
      // <init>
   }

   static {
      // <clinit>
   }

}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

clinit詳解

在準備階段,變數已經賦過一次系統要求的初始值,而在初始化階段,則根據程式設計師通過程式制定的主觀計劃去初始化類變數和其他資源,或者可以從另外一個角度來表達:初始化階段是執行類構造器<clinit>()方法的過程。

①<clinit>()方法是由編譯器自動收集類中的所有類變數的賦值動作和靜態語句塊(static{}塊)中的語句合併產生的,編譯器收集的順序是由語句在原始檔中出現的順序所決定的,靜態語句塊中只能訪問到定義在靜態語句塊之前的變數,定義在它之後的變數,在前面的靜態語句塊可以賦值,但是不能訪問如下程式碼

public class Test{
static{
i=0;//給變數賦值可以正常編譯通過
System.out.print(i);//這句編譯器會提示"非法向前引用"
}
static int i=1;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

虛擬機器會保證在子類的<clinit>()方法執行之前,父類的<clinit>()方法已經執行完畢。 因此在虛擬機器中第一個被執行的<clinit>()方法的類肯定是java.lang.Object。由於父類的<clinit>()方法先執行,也就意味著父類中定義的靜態語句塊要優先於子類的變數賦值操作,如下程式碼中,欄位B的值將會是2而不是1。

static class Parent{
    public static int A=1;
    static{
    A=2;}
    static class Sub extends Parent{
    public static int B=A;
    }
    public static void main(String[]args){
    System.out.println(Sub.B);
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

介面中不能使用靜態語句塊,但仍然有變數初始化的賦值操作,因此介面與類一樣都會生成<clinit>()方法。 但介面與類不同的是,執行介面的<clinit>()方法不需要先執行父介面的<clinit>()方法。 只有當父介面中定義的變數使用時,父接口才會初始化。 另外,介面的實現類在初始化時也一樣不會執行介面的<clinit>()方法。 
注意:介面中的屬性都是static final型別的常量,因此在準備階段就已經初始化話。