java類加載機制-類定義中new如何理解
問題描述
問題解答
回答1:在運行 java 程序的時候,每個類只被加載一次。不可能重復加載的。所以也就不存在題主說的遞歸了。
回答2:推薦先了解一下JVM虛擬機中內存
AVA對象實例化過程中,主要使用到的包括虛擬機棧,JAVA堆和方法區。 JAVA文件經編譯之后首先會被加到到JVM方法區,JVM方法區中很重要的一個部分是運行時常量池——用以存儲class文件類的版本、字段、方法、接口等描述信息和編譯期間的常量和靜態變量。 JAVA對象真正進行實例化的地方在JAVA堆和虛擬機棧中,Object A = New Object();在實際內存中,A其實相當于我們給Ojbect這個類的實現起的一個名字,在面向對象編程中,就像狗是屬于一類動物,但是特指的那一條狗我們會給他起一個名字用以區分一樣。Object用以標記A是屬于這個類,而A是特指Object的一個具體實現,而New Object就相當于對這個類創建一個具體實現。所以我們可以了解到,一個對象他首先必須可以指明所屬的類,其次它還必須能指明他所特指的哪一個具體實現。 對應的有兩種實現方式:
1.句柄訪問對象
2.直接指針訪問對象 直接指針訪問對象
HotSpot采用的是第二種實現方式。 Class的裝載包括3個步驟:加載(loading),連接(link),初始化(initialize) 加載 根據上圖所示,我們不難理解,當一個對象進行實例化的時候,JVM會根據所需對象類型在JAVA堆中劃分內存區,并生成指向方法區對象數據類型的指針用以標識對象。 鏈接 虛擬機棧中的本地變量表(也有稱為局部變量表)中指針指向JAVA堆中劃分好的內存區域。JAVA虛擬機采用動態鏈接方式,只有編譯后的class文件并未存儲最終方法在內存的表現形式。 初始化 初始化實際上是對class文件中的初始化方法進行調用,其核心還是虛擬機棧中棧幀的一次POP/PUSH。相當于對類中的對象進行一次同樣的裝載過程。 至此,一個對象完整的實例化過程就全部介紹完畢。
類內部初始化順序為 靜態域->非靜態域->構造方法
推薦看一下:【JAVA筆記——道】對象生命周期詳解【JAVA筆記——道】Class初始化理解
回答3:是不是把類加載和對象初始化搞混了?
回答4:樓主是不是想的太高深了?實例化和聲明定義是2個概念和階段。A2有幾個靜態的成員變量在聲明時實例化,而類的實例化過程是通過構造函數完成的。這里看上去出現了嵌套,其實不然。編譯器可以在編譯實例化語句的時候申明指令跳轉,跳轉地址在二次編譯的時候確定。這樣,類申明階段,在實例化a,b,c,d的時候編譯程序會跳轉到A2(int i)的構造方法程序塊。在類實例化階段的時候跳轉到A()構造方法。說白了就是在不同階段調用對應的構造方法而已,沒有樓主以為的循環嵌套。
回答5:類加載時,對于靜態變量和構造方法的加載順序不一樣吧。構造方法本質也是靜態方法。也不怎么懂,先占個板凳
相關文章:
1. 測試自動化html元素選擇器元素ID或DataAttribute [關閉]2. html5和Flash對抗是什么情況?3. 運行python程序時出現“應用程序發生異常”的內存錯誤?4. javascript - QQ第三方登錄的問題5. node.js - mongodb查找子對象的名稱為某個值的對象的方法6. 利用IPMI遠程安裝centos報錯!7. 在mac下出現了兩個docker環境8. spring-mvc - spring-session-redis HttpSessionListener失效9. java - Spring boot 讀取 放在 jar 包外的,log4j 配置文件,系統有創建日志文件,不寫入日志信息。10. 正在使用electron和node.js做桌面應用,需要實時監聽是否有網絡連接,node或者electron是否可以做到

網公網安備