前言
上一篇我們知道了一個類的生命周期是:加載->驗證->准備->解析->初始化->使用->卸載。
當初始化完成以后,一個類所有的類變量(被static修飾的變量)都被賦值。但是未被static修飾的成員變量又是何時被賦值的呢?
一個類何時會被初始化
一個類何時被初始化可以分為以下幾類:
1.創建類的實例(new)。
2.訪問某個類或接口的靜態變量,或者對該靜態變量賦值。
3.調用類的靜態方法。
4.通過反射方式執行以上三種行為。
5.初始化子類的時候,會觸發父類的初始化。
6.Java虛擬機啟動時被標明為啟動類的類。(有main方法的類)
7.JDK 1.7開始提供的動態語言支持。(了解即可)
我們來說道說道第3點和第6點
我們平常在使用main方法和調用某個類的靜態方法的時候,是不是發現,並不能直接調用靜態方法和main方法所在類的非靜態方法和非靜態變量。
可是明明不是說了在調用靜態方法和執行main方法的時候,所在的類已經被初始化了嗎?
是的!在上一章節我們就說,類初始化的時候會按照我們編寫代碼的順序為類變量(static修飾的變量)進行賦值。注意哦,此時這個類僅僅只有靜態變量被正確賦值了哦。
public class People{
private static String name ="lisi";
private int age = 18;
static{
name ="zhangsan";
}
}
如上述代碼,在該類被初始化之后,首先按照代碼順序將類變量(被static修飾的變量)賦值。所以name被賦值"lishi",然后再將name賦值為 "zhangsan"。此時age並沒有值,直接直接調用會報錯。
一個類何時被實例化
上一個章節中,我們明白了加載->驗證->准備->解析->初始化的具體細節。
當初始化完成后,一個類的靜態變量被正確賦值。如果這個對象是被new出來的。那么在初始化完成之后會進入實例化階段。
實例化的具體步驟為:
父類非靜態成員初始化語句(包括代碼塊,按照在類定義中的順序執行)->父類構造函數->子類非靜態成員初始化語句(包括代碼塊,按照在類定義中的順序執行)->子類構造方法()
注意哦!
如果這個類有父類不光是先實例化父類。整體流程如下:
1. 加載父類
1.1 為靜態屬性分配存儲空間並賦初始值
1.2 執行靜態初始化塊和靜態初始化語句(從上至下)
2. 加載子類
2.1 為靜態屬性分配存儲空間
2.2 執行靜態初始化塊和靜態初始化語句(從上至下)
3. 加載父類構造器
3.1 為實例屬性分配存數空間並賦初始值
3.2 執行實例初始化塊和實例初始化語句
3.3 執行構造器內容
4. 加載子類構造器
4.1 為實例屬性分配存數空間並賦初始值
4.2 執行實例初始化塊和實例初始化語句
4.3 執行構造器內容
實例化的內存模型
到這里,我們就非常的清楚為什么在靜態方法中不能直接調用非靜態的變量,因為此時的非靜態變量並沒有被賦值。
以前我在學習的時候,new一個對象出來大家總是在說,我們初始化了這個對象,然后balabala。。。
導致我對於初始化和實例化里的靜態變量和非靜態變量何時被賦值一直有點模糊。new一個對象出來准確的來說是進行了初始化和實例化兩個步驟,這樣應該就會清晰了很多。
初始化完成以后,類被存放在方法區,注意哦,此時並沒有存放在堆內存中。
只有當對象實例化進入堆內存中以后才會對非靜態變量進行初始化賦值。
總結
這下總算吧一個類的初始化和實例化的細節搞明白了,關於文中的方法區,堆內存等內容,下一節再具體分析。