今天小編給大家分享一下Java之JVM的知識點(diǎn)有哪些的相關(guān)知識點(diǎn),內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。
成都創(chuàng)新互聯(lián)公司2013年開創(chuàng)至今,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務(wù)公司,擁有項(xiàng)目成都網(wǎng)站設(shè)計(jì)、網(wǎng)站建設(shè)網(wǎng)站策劃,項(xiàng)目實(shí)施與項(xiàng)目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元廣德做網(wǎng)站,已為上家服務(wù),為廣德各地企業(yè)和個人服務(wù),聯(lián)系電話:028-86922220
JVM為什么要劃分出這些區(qū)域呢?JVM內(nèi)存是從操作系統(tǒng)里面申請過來的,而JVM就根據(jù)功能需求將這些劃分成了一些小的模塊,這樣一塊大的場地就可以劃分成一些小的模塊,然后每個模塊就負(fù)責(zé)自己的功能就可以了,那接下來看看這些區(qū)域的功能到底是什么呢!
程序計(jì)數(shù)器是內(nèi)存中最小的區(qū)域,這里面主要保存了下一條要執(zhí)行的指令的地址在哪里(指令就是字節(jié)碼,一般程序要運(yùn)行,JVM就需要把字節(jié)碼加載出來放到內(nèi)存中,然后程序再把一條一條的指令從內(nèi)存中取出來放到CPU上去執(zhí)行,所以必須要記住當(dāng)前執(zhí)行到哪一條指令,以及下一條在哪里,因?yàn)镃PU不是只給一個進(jìn)程提供服務(wù)的,是給所有的進(jìn)程都提供服務(wù),是并發(fā)式的執(zhí)行程序的,又因?yàn)椴僮飨到y(tǒng)是以線程為單位進(jìn)行調(diào)度執(zhí)行的,所以每個線程都要有自己的執(zhí)行位置,也就是每一個線程都需要有一個程序計(jì)數(shù)器來記錄位置!)
棧里面存放的主要是局部變量和方法調(diào)用信息,只要涉及到新方法的調(diào)用,就會有"入棧"的操作,每執(zhí)行完成一個方法,就會有"出棧"的操作,而且棧也是每個線程都有一份的
因此對于遞歸來說,一定要控制好遞歸條件,否則很有可能會出現(xiàn)棧溢出(StackOverflowException)異常的!
堆是內(nèi)存中空間最大的區(qū)域,而且堆是每個進(jìn)程只有一份的,進(jìn)程中的多個線程公用一個堆,里面主要存放著new出來的對象以及對象的成員變量,例如String s = new String()如果在方法里面這里的s就是局部變量是在棧上的,如果這個s是成員變量,就是在堆上的,而后面new String()是對象的本體,對象是在堆上的,這是容易混淆的地方,另外堆還有一個重要的點(diǎn)就是關(guān)于垃圾回收問題,這個后面再詳細(xì)介紹!
方法區(qū)中存放的是"類對象",平常所寫的.java代碼經(jīng)過編譯器翻譯過后就會變成.class(二進(jìn)制字節(jié)碼),然后.class就會被加載到內(nèi)存中,也就被JVM構(gòu)造成了類對象(加載的過程就是稱為"類加載"),而這些類對象就會存放到方法區(qū)中,這里面就具體描述了類長啥樣(類的名字,類的成員及其成員名成員類型,類的方法及其方法名方法類型,以及一些指令…另外類對象里面還存放了一個很重要的東西,就是靜態(tài)成員,一般被static修飾的成員就成為了類屬性,而普通的方法被稱為實(shí)例屬性,這是有很大差別的)!
上面所介紹的是JVM中比較常見的區(qū)域,而一些JVM的內(nèi)存區(qū)域劃分不一定是符合實(shí)際情況的,JVM在實(shí)現(xiàn)的過程中區(qū)域的劃分是不盡相同的,不同的廠商不同版本的JVM都是有可能存在差異的,不過對于我們普通的程序員而講,只要不是去實(shí)現(xiàn)JVM,那么就不需要了解那么深刻,講上面的幾個常見的區(qū)域加以了解就可以了!
類加載其實(shí)是設(shè)計(jì)一個運(yùn)行時環(huán)境的一個重要的功核心功能,這是非常重量級的,因此我這里也就簡單介紹一下!
上述就是類加載的具體過程,最后面的Using和Unloading就是使用的過程就不介紹了,就介紹一下前面的三個大的步驟:
在loading階段就會先找到對應(yīng)的.class文件,然后打開并讀取(根據(jù)字節(jié)流).class文件,同時初步生成一個類對象,這個和完成的類加載(class Loading)是不相同的,不要弄混淆了!
class文件的具體格式(如果要實(shí)現(xiàn)一個Java編譯器就得按照這樣的格式來構(gòu)造,實(shí)現(xiàn)JVM就得按照這個格式來進(jìn)行加載!):
觀察這個格式就可以看到.class文件就把.java文件中的核心信息都表述進(jìn)去了,只不過組織格式上發(fā)生了轉(zhuǎn)變,所以loading環(huán)節(jié)就會把讀取到的信息,初步填寫到類對象中
連接一般就是建立好多個實(shí)體之間的聯(lián)系
Verification就是一個校驗(yàn)的過程,主要就是驗(yàn)證讀到的內(nèi)容是不是和規(guī)范中規(guī)定的格式完全匹配,如果發(fā)現(xiàn)讀到的數(shù)據(jù)格式不符合規(guī)范,就會類加載失敗,并且拋出異常!
Preparation階段是正式為定義的變量(靜態(tài)變量,就是static修飾的變量)分配內(nèi)存并設(shè)置類變量初始值的階段,就會給每個靜態(tài)變量分配內(nèi)存,并且設(shè)置為0值!
Resolution階段是Java虛擬機(jī)將常量池內(nèi)的符號引用替換為直接引用的過程,也就是初始化常量的過程,.class文件中常量是集中放置的,每個常量會有一個編號,而在.class文件中的結(jié)構(gòu)體里初始情況就只是記錄的編號,然后就可以根據(jù)這個編號找到對應(yīng)的內(nèi)容,再填充到類對象中!
Initialization階段就是真正的對類對象進(jìn)行初始化(根據(jù)寫的代碼),尤其是針對靜態(tài)成員
class A { public A(){ System.out.println("A的構(gòu)造方法"); } { System.out.println("A的構(gòu)造代碼塊"); } static { System.out.println("A的靜態(tài)代碼塊"); }}class B extends A{ public B(){ System.out.println("B的構(gòu)造方法"); } { System.out.println("B的構(gòu)造代碼塊"); } static { System.out.println("B的靜態(tài)代碼塊"); }}public class Test extends B{ public static void main(String[] args) { new Test(); new Test(); }}
可以自己先嘗試寫一下輸出的結(jié)果
做這樣的題就需要把握幾個大的原則:
類加載階段就會進(jìn)行靜態(tài)代碼塊的執(zhí)行,要想創(chuàng)建實(shí)例,勢必要先進(jìn)行類加載
靜態(tài)代碼塊只是類加載階段執(zhí)行一次,其他階段都不會再執(zhí)行
構(gòu)造方法和構(gòu)造代碼塊每次實(shí)例化都會執(zhí)行,而且構(gòu)造代碼塊會在構(gòu)造方法前面執(zhí)行~~
父類執(zhí)行在前,子類執(zhí)行在后!
程序是從main開始執(zhí)行的,main的Test的方法,因此要執(zhí)行main就需要先加載Test類
只有涉及到這個類了,類里面的東西才會被加載
輸出結(jié)果: A的靜態(tài)代碼塊 B的靜態(tài)代碼塊 A的構(gòu)造代碼塊 A的構(gòu)造方法 B的構(gòu)造代碼塊 B的構(gòu)造方法 A的構(gòu)造代碼塊 A的構(gòu)造方法 B的構(gòu)造代碼塊 B的構(gòu)造方法
這個東西是類加載中的一個環(huán)節(jié),處于Loading階段(比較靠前的部分),雙親委派模型描述的就是JVM中的類加載器,如何根據(jù)類的全限定名(java.lang.String)找到.class文件的過程。這里的類加載器是JVM專門提供的對象,主要負(fù)責(zé)進(jìn)行類加載,所以找文件的過程也是由類加載器來負(fù)責(zé)的,.class文件可能放置的位置有很多,有的要放到JDK目錄里面,有的放到項(xiàng)目目錄里面,還有的在其他特定的位置里面,因此JVM提供了多個類加載器,每個類加載器負(fù)責(zé)一個片區(qū),而默認(rèn)的類加載器主要有3個:
BootStrapClassLoader:負(fù)責(zé)加載標(biāo)準(zhǔn)庫中的類(String,ArrayList,Random,Scanner…)
ExtensionClassLoader:負(fù)責(zé)加載JDK擴(kuò)展的類(現(xiàn)在很少用到)
ApplicationClassLoader:負(fù)責(zé)加載當(dāng)前項(xiàng)目目錄中的類
另外程序員還可以自定義類加載器,來加載其他目錄中的類,Tomcat就自定義了類加載器,用來專門加載webapps里面的.class
雙親委派模型就描述了這個找目錄的過程,也就是上述類加載器是如何配合的
考慮找一下java.lang.String:
程序啟動,就會先進(jìn)入ApplicationClassLoader類加載器
ApplicationClassLoader類加載器就會檢查下,它的父加載器是否已經(jīng)加載過了,如果沒有,就調(diào)用父 類加載器ExtensionClassLoader
ExtensionClassLoader類加載器就會檢查下,它的父加載器是否已經(jīng)加載過了,如果沒有,就調(diào)用父 類加載器BootStrapClassLoader
BootStrapClassLoader類加載器也會檢查下,它的父加載器是否已經(jīng)加載過了,然后發(fā)現(xiàn)沒有父親,于是就掃描自己負(fù)責(zé)的目錄
然后java.lang.String這個類就在標(biāo)準(zhǔn)庫中能找到,然后后續(xù)就由BootStrapClassLoader加載器負(fù)責(zé)后續(xù)的加載過程,查找環(huán)節(jié)就結(jié)束了!
考慮找一下自己寫的Test類:
程序啟動,就會先進(jìn)入ApplicationClassLoader類加載器
ApplicationClassLoader類加載器就會檢查下,它的父加載器是否已經(jīng)加載過了,如果沒有,就調(diào)用父 類加載器ExtensionClassLoader
ExtensionClassLoader類加載器就會檢查下,它的父加載器是否已經(jīng)加載過了,如果沒有,就調(diào)用父 類加載器BootStrapClassLoader
BootStrapClassLoader類加載器也會檢查下,它的父加載器是否已經(jīng)加載過了,然后發(fā)現(xiàn)沒有父親,于是就掃描自己負(fù)責(zé)的目錄,沒掃描到,就會回到子加載器中繼續(xù)掃描
ExtensionClassLoader掃描自己負(fù)責(zé)的目錄,也沒有掃描到,再回到子加載器中繼續(xù)掃描
ApplicationClassLoader也掃描自己負(fù)責(zé)的目錄,自己寫的類就在自己的項(xiàng)目目錄下,因此就能找到,然后后續(xù)的類加載就由ApplicationClassLoad完成,此時查找目錄的環(huán)節(jié)就結(jié)束了~~(另外如果ApplicationClassLoader也沒有找到們就會拋出ClassNotFoundException異常)
這一套查找規(guī)則就稱為雙親委派模型,那為啥JVM要這樣設(shè)計(jì)呢,理由就是一旦程序員自己寫的類和全限定類名重復(fù)了,也能夠成功加載標(biāo)準(zhǔn)庫中的類,而不是自己寫的類!!!
另外如果是自定義的類加載器,要不要遵守這個雙親委派模型呢?
答案是可以遵守也可以不遵守,主要看需求,例如Tomcat加載webapp中的類,就沒有遵守,因?yàn)樽袷亓松厦娴念惣虞d器也是不可能找到的!
JVM中的垃圾回收機(jī)制(GC),一般在寫代碼的時候,經(jīng)常就會涉及到申請內(nèi)存,例如創(chuàng)建一個變量,new一個對象,調(diào)用一個方法,加載類…而申請內(nèi)存的時機(jī)一般是明確的(需要保存某個或某些數(shù)據(jù)就需要申請內(nèi)存),但是釋放內(nèi)存的時機(jī),卻是不那么清楚的,釋放的早了也不行(如果還是要使用的,結(jié)果已經(jīng)被釋放了這就讓其無內(nèi)存可用了,就讓這些數(shù)據(jù)"無處可去"),釋放的晚了也不行(釋放晚了,大量的囤積很有可能讓可用內(nèi)存逐漸變少,很有可能會出現(xiàn)內(nèi)存泄漏問題,就是無內(nèi)存可以使用),因此內(nèi)存的釋放要恰到好處才好!
而垃圾回收的本職是靠運(yùn)行時環(huán)境額外做了很多的工作來完成釋放內(nèi)存操作的,這讓程序員的心智負(fù)擔(dān)大大降低了,但是垃圾回收也是有劣勢的:①消耗額外的開銷(消耗資源耕更多了);②可能會影響程序的流暢運(yùn)行(垃圾回收會經(jīng)常引入STW問題(Stop The World))
垃圾回收的內(nèi)存有哪些呢,是全部都要回收嘛?
當(dāng)然不是了,就用上面的四個區(qū)域來說一下:
程序計(jì)數(shù)器:這個內(nèi)存是固定大小的,不涉及到釋放,也就不需要GC了;
棧:當(dāng)函數(shù)調(diào)用完畢,對應(yīng)的棧幀也就自動釋放了,也是不需要GC的;
堆:這是最需要GC的內(nèi)存,一般代碼中的大量的內(nèi)存都在堆上;
而這三個區(qū)域到底哪些是需要釋放的,對于這種一部分在使用,一部分不再使用的對象,整體來說就是不釋放的,只有等到這個對象完全不再使用,才真正的進(jìn)行釋放,因此在GC中就不會出現(xiàn)半個對象的情況,因此垃圾回收的基本單位就是對象,而不是字節(jié)!方法區(qū):類對象,類加載的,而只有進(jìn)行到類卸載的時候才需要進(jìn)行釋放內(nèi)存,而卸載操作是非常低頻的,因此幾乎就不涉及到GC!
下面就具體來看一下是怎么回收的:
而當(dāng)下有兩個主流的方案:
這不是Java中采取的方案,這是Python及其他語言的方案,因此這里就簡單介紹一下,就不過多介紹了~
而引用計(jì)數(shù)的具體思路就是針對每個對象,都會額外引入一小塊內(nèi)存,來保存這個對象有多少個引用指向它
而這樣的引用計(jì)數(shù)存在兩個缺陷:
空間利用率比較低!!!,每個new的對象都需要搭配一個計(jì)數(shù)器,假設(shè)一個計(jì)數(shù)器4個字節(jié),如果對象本身比較大(幾百個字節(jié)),那么這個計(jì)數(shù)器就無所謂,而一旦這個對象本身就比較小(4個字節(jié)),那么再多出來4個字節(jié),就相當(dāng)于空間利用率就浪費(fèi)了一倍,因此空間利用率會比較低~
有循環(huán)引用的問題
因此使用引用計(jì)數(shù)也是會有大量的問題出現(xiàn)的,而想Python,PHP之類的語言也不是只使用引用計(jì)數(shù)器就完成GC的,也是配合了一些其他的機(jī)制來完成的!
可達(dá)性分析是Java所采取的方案,可達(dá)性分析是通過一些額外的線程,定期針對整個內(nèi)存空間的對象進(jìn)行掃描,有一些起始位置(GCRoots),然后就類似于深度優(yōu)先遍歷一樣(可以想象成是一棵樹),把可以訪問到的對象都標(biāo)記一邊(帶有標(biāo)記的對象就是可達(dá)的對象),而沒有被標(biāo)記的對象,就是不可達(dá)的對象,也就是垃圾,應(yīng)該被釋放掉!
這里的GCRoots(從這些位置開始遍歷):
棧上的局部變量;
常量池中的引用指向的對象;
方法區(qū)中的靜態(tài)成員指向的對象;
因此可達(dá)性分析的優(yōu)點(diǎn)就是解決了引用計(jì)數(shù)的缺點(diǎn):空間利用率低,循環(huán)引用;而可達(dá)性分析的缺點(diǎn)也很明顯:系統(tǒng)開銷大,遍歷一次可能比較慢~
因此找垃圾也是很簡單的,核心就是確認(rèn)這個對象未來是否還會使用,看還有沒有引用指向它,應(yīng)不應(yīng)該釋放掉!
既然已經(jīng)明確了什么是垃圾,接下來就要回收垃圾了,而回收垃圾有三種基本策略,下面來看一下!
這里的標(biāo)記就是可達(dá)性分析的過程,而清除就是釋放內(nèi)存,假設(shè)上面是一塊內(nèi)存,而打鉤的區(qū)域代表是垃圾,此時如果直接釋放掉,雖然內(nèi)存是還給系統(tǒng)了,但是釋放掉的內(nèi)存是離散的,不是連續(xù)的,而這樣帶來的問題就是"內(nèi)存碎片",空閑的內(nèi)存可能會有很多,假設(shè)加起來一共是1G,而此時想要申請500MB的空間,按理是可以申請到的,但在這里是有可能申請失敗的(因?yàn)橐暾埖?00MB是連續(xù)的內(nèi)存,每次申請的內(nèi)存都是連續(xù)的內(nèi)存空間,而這里的1G可能是多個碎片加起來的),因此這樣的問題其實(shí)是非常影響程序運(yùn)行的
由于上面的標(biāo)記-清除策略可能會帶來內(nèi)存碎片的問題,因此引入了復(fù)制算法來解決這一問題
上面是一塊內(nèi)存,復(fù)制算法的策略就是內(nèi)存使用一半,丟一半,不全部使用,在使用的一般里面把不是垃圾的拷貝到另一半(這個拷貝是JVM內(nèi)部處理好的,不用糾結(jié)),然后把前面使用的全部內(nèi)存都釋放掉,這樣內(nèi)存碎片的問題就迎刃而解了!
所以復(fù)制算法就有兩個很大的問題:
內(nèi)存空間利用率低(只使用了一般的內(nèi)存);
如果要保留的對象多,要釋放的對象少,那么復(fù)制的開銷就很大;
這又是針對復(fù)制算法,再進(jìn)一步做出改進(jìn)!
標(biāo)記整理的策略就是將不是垃圾的內(nèi)存整理到一起,然后釋放掉后面的全部內(nèi)存,就類似于順序表刪除中間元素的操作一樣,有一個搬運(yùn)的過程!
這個方案空間利用率是高了,但是仍然沒有辦法解決復(fù)制/搬運(yùn)元素開銷大的問題!
上述的三種方案,雖然能夠解決問題,但是都有各自的缺陷,因此實(shí)際上JVM中的實(shí)現(xiàn),會把多種方案結(jié)合起來使用,也就是"分代回收"!!!
這里的分代就是針對對象來進(jìn)行分類(根據(jù)對象的"年齡"進(jìn)行分類,而這里的年齡表示一個對象熬過一輪GC的掃描,就稱"長了一歲"),而針對不同年齡的對象,就采取不同的方案!!!
這就是整個分代回收的過程!
上面的找垃圾和釋放垃圾都只是算法的思想,并不是真正的落地實(shí)現(xiàn)的過程,而真正實(shí)現(xiàn)上述算法模塊的是"垃圾回收器",下面來介紹一些具體的垃圾回收器:
Serial收集器是給新生代提供的垃圾回收器,Serial Old收集器是給老年代提供的垃圾回收器,這兩個收集器是串行收集的,而且在進(jìn)行垃圾的掃描和釋放的時候,業(yè)務(wù)線程要停止工作,所以這樣的方式掃描的滿,釋放的也慢,而且也能產(chǎn)生嚴(yán)重的STW!
ParNew收集器,Parallel Scavenge收集器都是提供給新生代的,Parallel Scavenge收集器比起ParNew收集器加了一些參數(shù),可以控制STW的時間,就是多了一些更強(qiáng)的功能,Parallel Old收集器是提供給老年代的,這三個收集器都是并行收集的,就是引入了多線程的方式來解決掃描垃圾和釋放垃圾的功能!
上面的這幾個回收器都是歷史遺留下來的,也就是比較老的垃圾回收方式,另外再介紹兩個更新的垃圾回收器!
CMS收集器設(shè)計(jì)的比較巧妙,其設(shè)計(jì)的初衷是盡可能讓STW時間短,Java8使用的正是CMS收集器,下面簡單介紹一下CMS收集器的過程:
初始標(biāo)記:速度很快,會引起短暫的STW(只是找到GCRoots);
并發(fā)標(biāo)記:速度很快,但是可以和業(yè)務(wù)線程并發(fā)執(zhí)行,不會產(chǎn)生STW;
重新標(biāo)記:在2業(yè)務(wù)代碼可能會影響并發(fā)標(biāo)記的結(jié)果(業(yè)務(wù)線程在執(zhí)行,就有可能產(chǎn)生新的垃圾),因此這一步就是針對2的結(jié)果進(jìn)行微調(diào),雖然會引起STW,但只是微調(diào),速度很快;
上面三步都是基于可達(dá)性分析!
回收內(nèi)存:也是和業(yè)務(wù)線程并發(fā)執(zhí)行,不會產(chǎn)生STW,這是基于標(biāo)記整理;
G1收集器是唯一一款全區(qū)域的垃圾回收器,從Java11開始使用的就是G1收集器,這個收集器是把整個內(nèi)存,分成了很多小的區(qū)域Region,給這些Region進(jìn)行了不同的標(biāo)記,有的Region放新生代對象,有的Region放老年代對象,然后掃描的時候,就一次掃描若干個Region(不追求一輪GC就掃描完,需要分多次掃描),這樣對于業(yè)務(wù)代碼的影響也是最小的,
這兩個新的收集器的核心思想就是化整為零,G1當(dāng)下可以優(yōu)化到讓STW停頓時間小于1ms,這是完全可以接收的!上面就是關(guān)于JVM的一些學(xué)習(xí)了,這里的收集器主要還是了解為主,主要還是上面的垃圾回收思想很重要!!!
以上就是“Java之JVM的知識點(diǎn)有哪些”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習(xí)更多的知識,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。
當(dāng)前題目:Java之JVM的知識點(diǎn)有哪些
分享地址:http://aaarwkj.com/article36/pjsosg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供Google、定制網(wǎng)站、云服務(wù)器、建站公司、響應(yīng)式網(wǎng)站、網(wǎng)站排名
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)