衆所周知,在java裏是不能給構造函數寫返回值的,若是在低版本的編譯器定義一個構造器寫上返回值可能會報錯,高版本里面他就是一個普通的方法。
但是若是構造函數沒有返回值,那麼好比Test t = new Test()咱們new一個對象的時候是怎麼賦值的呢?java
寫一段代碼測試一下:函數
public class Test { public Test() { } public static void main(String[] args) { Test t = new Test(); } }
反編譯一下看看:測試
Code: 0: new #5 // class com/irving/utils/baidu/Test 3: dup 4: invokespecial #6 // Method "<init>":()V 7: astore_1 8: return
從反編譯的結果看 4: invokespecial #7 // Method "init": ()V,調用構造函數,V表明void無返回值,那麼init表明什麼含義?this
我在書裏找到這樣一段話:編碼
在 Java 虛擬機層面上,Java 語言中的構造函數是以一個名爲init的特殊實例初始化方法的形式出現的,init這個方法名稱是由編譯器命名的,由於它並不是一個合法的 Java 方法名字,不可能經過程序編碼的方式實現。實例初始化方法只能在實例的初始化期間,經過 Java 虛擬機的 invokespecial 指令來調用, 只有在實例正在構造的時候,實例初始化方法才能夠被調用訪問。一個類或者接口最多能夠包含不超過一個類或接口的初始化方法,類或者接口就是經過這個方法完成初始化的。這個方法是一個不包含參數的靜態方法,名爲clinit。這個名字也是由編譯器命名的,由於它並不是一個合法的 Java 方法名字,不可能經過程序編碼的方式實現。類或接口的初始化方法由 Java 虛擬機自身隱式調用,沒有任何虛擬機字節碼指令能夠調用這個方法,只有在類的初始化階段中會被虛擬機自身調用。spa
init表明着虛擬機調用構造函數,如今狀況很明顯,構造函數返回類型是void,那麼它到底是怎麼賦值的呢?code
咱們明白一點,方法的調用過程就是棧幀入棧和出棧的過程,棧幀隨着方法的調用建立,方法結束銷燬。棧幀的內部包含局部變量表、操做數棧、動態連接等。對象
局部變量表表示方法調用時候的參數傳遞,當一個實例方法被調用的時候,第0個局部變量存儲了當前實例方法所在對象的引用(this),後續的其餘參數傳遞至1到N的連續位置。索引
操做數棧用來準備方法調用的參數和返回結果。接口
以上面測試代碼的方法來看Test t = new Test() 的調用過程:
從這個過程咱們已經看出來了,整個過程最後咱們最終拿到了new以後建立的對象引用,而且保存到局部變量表中,能夠供咱們繼續使用。
關注公衆號:java寶典