如圖:
1.加載
我們編寫一個java的源文件,經過編譯后生成一個后綴名為.class的文件,這結合四字節碼文件,
java虛擬機就識別這種文件,java的生命周期就是class文件從加載到消亡的過程。
關於加載,其實,就是將源文件的class文件找到類的信息將其加載到方法區中,
然后在堆區中實例化一個java.lang.Class對象,作為方法區中這個類的信息的入口。
但是這一功能是在JVM之外實現的,主要的原因是方便讓應用程序自己決定如何獲取這個類,
在不同的虛擬機實現的方式不一定相同,hotspot虛擬機是采用需要時在加載的方式,
也有其他是先預先加載的。
2.連接
一般會跟加載階段和初始化階段交叉進行,過程由三部分組成:驗證、准備和解析三步
(1)驗證:確定該類是否符合java語言的規范,有沒有屬性和行為的重復,繼承是否合理,總之,就是保證jvm能夠執行
(2)准備:主要做的就是為由static修飾的成員變量分配內存,並設置默認的初始值
默認初始值如下:
1.八種基本數據類型默認的初始值是0
2.引用類型默認的初始值是null
3.有static final修飾的會直接賦值,例如:static final int x=10;則默認就是10.
(3)解析:這一階段的任務就是把常量池中的符號引用轉換為直接引用,說白了就是jvm會將所有的類或接口名、字段名、方法名轉換為具體的內存地址。
3.初始化
這個階段就是將靜態變量(類變量)賦值的過程,即只有static修飾的才能被初始化,執行的順序就是:
父類靜態域或着靜態代碼塊,然后是子類靜態域或者子類靜態代碼塊
4.使用
在類的使用過程中依然存在三步:對象實例化、垃圾收集、對象終結
(1)對象實例化:就是執行類中構造函數的內容,如果該類存在父類JVM會通過顯示或者隱示的方式先執行父類的構造函數,在堆內存中為父類的實例變量開辟空間,並賦予默認的初始值,然后在根據構造函數的代碼內容將真正的值賦予實例變量本身,然后,引用變量獲取對象的首地址,通過操作對象來調用實例變量和方法
(2)垃圾收集:當對象不再被引用的時候,就會被虛擬機標上特別的垃圾記號,在堆中等待GC回收
(3)對象的終結:對象被GC回收后,對象就不再存在,對象的生命也就走到了盡頭
5.類卸載
即類的生命周期走到了最后一步,程序中不再有該類的引用,該類也就會被JVM執行垃圾回收,從此生命結束…
package com.etc.test; class A{ static int a;//類變量 String name; int id; //靜態代碼塊 static{ a=10; System.out.println("這是父類的靜態代碼塊"+a); } //構造代碼塊 { id=11; System.out.println("這是父類的構造代碼塊id:"+id); } A(){ System.out.println("這是父類的無參構造函數"); } A(String name){ System.out.println("這是父類的name"+name); } } class B extends A{ String name; static int b; static{ b=12; System.out.println("這是子類的靜態代碼塊"+b); } B(String name) { super(); this.name = name; System.out.println("這是子類的name:"+name); } } public class Test666 { public static void main(String[] args) { B bb=new B("GG"); } }
輸出的結果如下:
這是父類的靜態代碼塊10
這是子類的靜態代碼塊12
這是父類的構造代碼塊id:11
這是父類的無參構造函數
這是子類的name:GG