前言:
站在用戶的角度思考問題,與客戶深入溝通,找到永康網站設計與永康網站推廣的解決方案,憑借多年的經驗,讓設計與互聯(lián)網技術結合,創(chuàng)造個性化、用戶體驗好的作品,建站類型包括:成都網站設計、成都做網站、企業(yè)官網、英文網站、手機端網站、網站推廣、域名申請、虛擬主機、企業(yè)郵箱。業(yè)務覆蓋永康地區(qū)。JVM是Java Virtual Machine(Java虛擬機)的縮寫,JVM是一種用于計算設備的規(guī)范,它是一個虛構出來的計算機,是通過在實際的計算機上仿真模擬各種計算機功能來實現的。引入Java語言虛擬機后,Java語言在不同平臺上運行時不需要重新編譯。Java語言使用Java虛擬機屏蔽了與具體平臺相關的信息,使得Java語言編譯程序只需生成在Java虛擬機上運行的目標代碼(字節(jié)碼),就可以在多種平臺上不加修改地運行。
下面,大家可以嘗試著回答一下以下問題:
1、JVM管理的內存結構是怎樣的?
2、不同的虛擬機在實現運行時內存的時候有什么區(qū)別?
3、運行時數據區(qū)中哪些區(qū)域是線程共享的?哪些是獨享的?
4、除了JVM運行時內存以外,還有什么區(qū)域可以用嗎?
5、堆和棧的區(qū)別是什么?
6、Java中的數組是存儲在堆上還是棧上的?
7、Java中的對象創(chuàng)建有多少種方式?
8、Java中對象創(chuàng)建的過程是怎么樣的?
9、Java中的對象一定在堆上分配內存嗎?
10、如何獲取堆和棧的dump文件?
以上10道題,如果您可以全部準確無誤的回答的話,那說明你真的很了解JVM的內存結構以及內存分配相關的知識了,如果有哪些知識點是不了解的,那么本文正好可以幫你答疑解惑。
1:JVM管理的內存結構是怎樣的?
Java虛擬機在執(zhí)行Java程序的過程中會把他所管理的內存劃分為若干個不同的數據區(qū)域?!禞ava虛擬機規(guī)范》中規(guī)定了JVM所管理的內存需要包括一下幾個運行時區(qū)域:
Java虛擬機運行時數據區(qū)域主要包含了PC寄存器(程序計數器)、Java虛擬機棧、本地方法棧、Java堆、方法區(qū)以及運行時常量池。
各個區(qū)域有各自不同的作用,關于各個區(qū)域的作用就不在本文中相信介紹了。
但是,需要注意的是,上面的區(qū)域劃分只是邏輯區(qū)域,規(guī)范對于有些區(qū)域的限制是比較松的,所以不同的虛擬機廠商在實現上,甚至是同一款虛擬機的不同版本也是不盡相同的。
2:不同的虛擬機在實現運行時內存的時候有什么區(qū)別?
前面提到過《Java虛擬機規(guī)范》定義的JVM運行時所需的內存區(qū)域,不同的虛擬機實現上有所不同,而在這么多區(qū)域中,規(guī)范對于方法區(qū)的管理是最寬松的,規(guī)范中關于這部分的描述如下:
方法區(qū)在虛擬機啟動的時候創(chuàng)建,雖然方法區(qū)是堆的邏輯組成部分,但是簡單的虛擬機實現可以選擇在這個區(qū)域不實現垃圾收集與壓縮。本版本的規(guī)范也不限定實現方法區(qū)的內存位置和代碼編譯的管理策略。方法區(qū)的容量可以是固定的,也可以隨著程序執(zhí)行的需求動態(tài)擴展,并在不需要過多的空間時自行收縮。方法區(qū)在實際內存空間站可以是不連續(xù)的。
這一規(guī)定,可以說是給了虛擬機廠商很大的自由。
虛擬機規(guī)范對方法區(qū)實現的位置并沒有明確要求,在最著名的HotSopt虛擬機實現中(在Java 8 之前),方法區(qū)僅是邏輯上的獨立區(qū)域,在物理上并沒有獨立于堆而存在,而是位于永久代中。所以,這時候方法區(qū)也是可以被垃圾回收的。
實踐證明,JVM中存在著大量的聲明短暫的對象,還有一些生命周期比較長的對象。為了對他們采用不同的收集策略,采用了分代收集算法,所以HotSpot虛擬機把的根據對象的年齡不同,把堆分位新生代、老年代和永久代。
在Java 8中 ,HotSpot虛擬機移除了永久代,使用本地內存來存儲類元數據信息并稱之為:元空間(Metaspace)
?
3:運行時數據區(qū)中哪些區(qū)域是線程共享的?哪些是獨享的?
在JVM運行時內存區(qū)域中,PC寄存器、虛擬機棧和本地方法棧是線程獨享的。
而Java堆、方法區(qū)是線程共享的。但是值得注意的是,Java堆其實還未每一個線程單獨分配了一塊TLAB空間,這部分空間在分配時是線程獨享的,在使用時是線程共享的。
4:除了JVM運行時內存以外,還有什么區(qū)域可以用嗎?
除了我們前面介紹的虛擬機運行時數據區(qū)以外,還有一部分內存也被頻繁使用,他不是運行時數據區(qū)的一部分,也不是Java虛擬機規(guī)范中定義的內存區(qū)域,他就是——直接內存。
直接內存的分配不受Java堆大小的限制,但是他還是會收到服務器總內存的影響。
在JDK 1.4中引入的NIO中,引入了一種基于Channel和Buffer的I/O方式,他可以使用Native函數直接分配堆外內存,然后通過一個存儲在Java堆中的DirectByteBuffer對象作為這塊內存的應用進行操作。
?
5:堆和棧的區(qū)別是什么?
堆和棧(虛擬機棧)是完全不同的兩塊內存區(qū)域,一個是線程獨享的,一個是線程共享的,二者之間大的區(qū)別就是存儲的內容不同:
堆中主要存放對象實例。
棧(局部變量表)中主要存放各種基本數據類型、對象的引用。
6:Java中的數組是存儲在堆上還是棧上的?
在Java中,數組同樣是一個對象,所以對象在內存中如何存放同樣適用于數組;
所以,數組的實例是保存在堆中,而數組的引用是保存在棧上的。
?
Q7:Java中的對象創(chuàng)建有多少種方式?
Java中共有5種方式可以創(chuàng)建一個對象。
最簡單的方式就是使用new關鍵字。
User user = new User();
除此以外,還可以使用反射機制創(chuàng)建對象:
User user = User.class.newInstance();
或者使用Constructor類的newInstance:
Constructorconstructor=User.class.getConstructor();User user =constructor.newInstance();
除此之外還可以使用clone方法和反序列化的方式,這兩種方式不常用并且代碼比較復雜,就不在這里展示了,感興趣的可以自行了解下。
8:Java中對象創(chuàng)建的過程是怎么樣的?
對于一個普通的Java對象的創(chuàng)建,大致過程如下:
1、虛擬機遇到new指令,到常量池定位到這個類的符號引用。
2、檢查符號引用代表的類是否被加載、解析、初始化過。
3、虛擬機為對象分配內存。
4、虛擬機將分配到的內存空間都初始化為零值。
5、虛擬機對對象進行必要的設置。
6、執(zhí)行方法,成員變量進行初始化。
9:Java中的對象一定在堆上分配內存嗎?
前面我們說過,Java堆中主要保存了對象實例,但是,隨著JIT編譯期的發(fā)展與逃逸分析技術逐漸成熟,棧上分配、標量替換優(yōu)化技術將會導致一些微妙的變化,所有的對象都分配到堆上也漸漸變得不那么“絕對”了。
其實,在編譯期間,JIT會對代碼做很多優(yōu)化。其中有一部分優(yōu)化的目的就是減少內存堆分配壓力,其中一種重要的技術叫做逃逸分析。
如果JIT經過逃逸分析,發(fā)現有些對象沒有逃逸出方法,那么有可能堆內存分配會被優(yōu)化成棧內存分配。
10:怎么如何獲取堆和棧的dump文件?
Java Dump,Java虛擬機的運行時快照。將Java虛擬機運行時的狀態(tài)和信息保存到文件。
可以使用在服務器上使用jmap命令來獲取堆dump,使用jstack命令來獲取線程的調用棧dump。
讀者福利
針對于上面的面試題我總結出了互聯(lián)網公司java程序員面試涉及到的絕大部分面試題及答案做成了文檔和架構視頻資料免費分享給大家(包括Dubbo、Redis、Netty、zookeeper、Spring cloud、分布式、高并發(fā)等架構技術資料),希望能幫助到您面試前的復習且找到一個好的工作,也節(jié)省大家在網上搜索資料的時間來學習。合理利用自己每一分每一秒的時間來學習提升自己,不要再用"沒有時間“來掩飾自己思想上的懶惰!
獲取方式:資料整理不易,轉×××某人 免費領取。
另外有需要云服務器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內外云服務器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務器、裸金屬服務器、高防服務器、香港服務器、美國服務器、虛擬主機、免備案服務器”等云主機租用服務以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應用場景需求。