JVM思考-init和clinit區別


JVM思考-init和clinit區別

目錄:JVM總括:目錄


clinit和init的區別其實也就是Class對象初始化對象初始化的區別,詳情看我上一篇博客:

 JVM總括四-類加載過程、雙親委派模型、對象實例化過程

一、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 {
      // <clinit>
   }

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

   private int x = 1;   // <init>

   X(){
      // <init>
   }
   
}

三、clinit詳解

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

1、

  <clinit>()方法是由編譯器自動收集類中的所有類變量的賦值動作和靜態語句塊(static{}塊)中的語句合並產生的,編譯器收集的順序是由語句在源文件中出現的順序所決定的,靜態語句塊中只能訪問到定義在靜態語句塊之前的變量,定義在它之后的變量,在前面的靜態語句塊可以賦值,但是不能訪問如下代碼:

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

2、

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

static class Parent{
    public static int A=1static{
    A=2;
    }
    
    static class Sub extends Parent{
        public static int B=A;
    }
    public static void main(String[]args){
        System.out.println(Sub.B);
    }
}

3、

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

轉載:

https://blog.csdn.net/u013309870/article/details/72975536

 


免責聲明!

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



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