日本黄色一级经典视频|伊人久久精品视频|亚洲黄色色周成人视频九九九|av免费网址黄色小短片|黄色Av无码亚洲成年人|亚洲1区2区3区无码|真人黄片免费观看|无码一级小说欧美日免费三级|日韩中文字幕91在线看|精品久久久无码中文字幕边打电话

當前位置:首頁 > > 架構師社區(qū)
[導讀]學過 JVM的人都知道在JVM中在執(zhí)行Java程序的過程中會把JVM管理的內存進行劃分,叫做 運行時數據區(qū)。 JVM中管理的內存主要分為以下五個部分: 方法區(qū)(Method Area): Java堆(Heap):Java堆是JVM管理內存中最大的一塊區(qū)域,幾乎所有的Java對象的內存都在這


學過 JVM的人都知道在JVM中在執(zhí)行Java程序的過程中會把JVM管理的內存進行劃分,叫做 運行時數據區(qū)。

JVM中管理的內存主要分為以下五個部分:

  1. 方法區(qū)Method Area):

  2. Java堆Heap):Java堆是JVM管理內存中最大的一塊區(qū)域,幾乎所有的Java對象的內存都在這里分配,此區(qū)域也是GC最活躍的區(qū)域。

  3. 虛擬機棧VM Stack):此區(qū)域就是我們通常所說的“”,當線程執(zhí)行方法時會在此區(qū)域創(chuàng)建一個棧幀,用于儲存局部變量表動態(tài)鏈接、方法出口等信息。

  4. 本地方法棧Native Method Stack):本地方法棧虛擬機棧類似,只不過本地方法棧執(zhí)行的是Native方法服務。

  5. 程序計數器Program Counter Register):是當前線程執(zhí)行字節(jié)碼的行號指令器,是線程私有的。

當執(zhí)行一個Java方法,程序計數器記錄的是JVM正在執(zhí)行的字節(jié)碼指令,若是執(zhí)行的是本地方法,程序計數器為空。

你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <br>這五個部分都有自己的功能,有些區(qū)域是線程私有的,會跟隨著線程的創(chuàng)建和銷毀,有些區(qū)域是跟隨進程的啟動而存在,各自承擔著自己的職責。
   <p style=在方法區(qū)中還有一塊區(qū)域就是常量池,很多人叫它為永生代 PermGen(JDK 1.7的說法)。

隨著JDK8的到來,JVM不再有PermGen。但類的元數據信息還在,不再是存儲在連續(xù)的堆空間上,而是移動到叫做Metaspace(元空間)的本地內存(Native memory)中。

我們知道當我們執(zhí)行創(chuàng)建對象的時候,就會初始化對象的屬性信息,例如執(zhí)行如下的代碼:

public class TestObj {

    private int n=1;

    public static void main(String[] args) throws Exception {

        TestObj obj =new TestObj();

    }
}

該代碼非常的簡單,在main方法中執(zhí)行TestObj obj =new TestObj()就會初始化該對象的成員變量n為1,我們都知道這個過程叫做初始化,在初始化的時候也會進行半始化。

那么什么叫做半初始化呢?

半初始化就是當一個成員變量還未初始化為它真正的值,會先初始化為它默認是的值,例如int會先初始化為0,boolean會先初始化為false等。

對于半初始化,真正的講清楚要從執(zhí)行的字節(jié)碼指令分析,下面我們通過字節(jié)碼指令進行深入的分析,一個對象的初始化過程。

具體在idea中查看字節(jié)碼指令的方法可以自行百度,這個不難,通過idea中可以查看上面代碼執(zhí)行的指令,如下所示:

你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <br>在上面mian方法中執(zhí)行完,也就是對應這五條指令,如下所示:
  </section>
  <section style=
  1. new指令:表示首先在堆中申請一塊內存,此時堆中的內存中存儲著該對象屬性n的半初始化狀態(tài)值n=0。

  2. dup指令:表示復制引用。

  3. invokespecial指令:表示調用對象的初始化方法,后面對應的注釋Method " " ,此時屬性值n才會被初始化為1。

  4. astore_1指令:此時會將TestObj obj =new TestObj()的引用obj 與該堆中的對象建立連接。

  5. return指令:執(zhí)行完最后返回。

從上面的指令中分析可以看出,當創(chuàng)建一個對象的時候,主要分為以下三個步驟,執(zhí)行的原理圖如下:

你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <br>
   <strong style=了解完對象的半初始化,那么什么又是對象分配?

說到JVM中的對象分配,我們得從對象在JVM中執(zhí)行new指令后開始講起。客觀且慢,請聽我詳細道來。

在JVM中當遇到一條new指令時,會首先檢查這條指令的參數是否在常量池中能定位到一個類的符號引用,若是定位不到,就表示沒有被加載、解析和初始化過,就會先執(zhí)行加載該類。

JVM中加載類信息的詳細過程,請參考這一篇文章[面試官:你知道java類是怎么跑起來的嗎?問的我一臉懵]。

若是存在該符號引用表示之前已經加載過該類信息,接下來就直接執(zhí)行在堆中進行對象內存的分配。

但是隨著JVM的發(fā)展,JIT編譯器的出現,所有的對象分配在堆中就不那么絕對了,當創(chuàng)建對象為對象分配內存時,也會嘗試在棧上分配,在JVM書籍中的描述如下所示:

你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <br>
   <strong style=那么什么是逃逸技術? 每個線程執(zhí)行方法都會創(chuàng)建一個棧幀,該棧幀用于存儲方法的局部變量,當一個變量不會在其他方法中使用到,只在該方法中使用,就不會逃逸。

什么又是變量替換呢? 標量替換就是創(chuàng)建一個對象的時候,直接以對象的屬性進行入棧存儲,方法結束后直接彈棧結束,不會有GC的介入。

因此,在棧上分配是對JVM的一種優(yōu)化措施,減少了GC的活動,提高了Java虛擬機的執(zhí)行效率。

當對象執(zhí)行在堆上進行內存分配的時候,為了防止多線程分配內存存在混亂的情況,通常在多線程的時候對對象內存的分配    有以下兩種方案進行解決:

  1. 對分配內存的動作進行同步,但是同步的的操作太消耗性能,大大降低了JVM的性能。

  2. 對堆內存為每一個線程劃分一塊本地線程分配緩沖TLAB),是線程私有的,這樣每一個線程只需要在自己的TLAB中進行分配即可,就不用進行同步,也能達到線程安全的目的:

那么當一個對象在堆中分配完一個內存后,對象在堆中又是怎么存在的呢?

客觀不急請聽我慢慢道來,當對象在堆中進行完內存分配后,一個普通對象在堆中以如下圖的形式存在:

你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <ol style=
  • markword:存儲的是鎖信息、對象的年齡信息等。

  • class pointer:類型指針指向該對象class類型。

  • instance data:實例數據,儲存對象實例的數據值。

  • padding:對其若是該對象的大小不能被8字節(jié)整除,就會自動補齊為8字節(jié)的整數倍。

  • 那么對象都已經存在堆中了,我們又是怎么訪問該對象的?

    若要訪問堆中已經存在的對象,有以下兩種方式:

    (1) 句柄的方式:會在堆中劃分一塊下的內存作為句柄池,對象的引用不會直接存儲數據的地址,而是指向句柄池的指針,由句柄池的指針存儲數據的地址。

    句柄池的方式,由于對象引用不會直接指向數據的地址,這樣當GC進行回收垃圾的時候,移動對象,對象的地址改變了就不用改變reference的本身內容。

    這個也是句柄訪問方式的唯一優(yōu)點,具體句柄訪問方式的原理圖如下所示:

    你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <p style=(2) 直接方式:直接方式是reference直接指向數據的,這樣減少了一次指針的定位,速度快,直接訪問的方式原理圖如下:

    你的對象在哪里?長什么樣?我?guī)闳タ匆豢? >
   </figure>
   <blockquote style=

    注意: 在HostSpot的源碼實現中,使用的是第二種直接訪問的方式

    特別推薦一個分享架構+算法的優(yōu)質內容,還沒關注的小伙伴,可以長按關注一下:

    你的對象在哪里?長什么樣?我?guī)闳タ匆豢? ></p><pre ng-bind-html=

    長按訂閱更多精彩▼

    你的對象在哪里?長什么樣?我?guī)闳タ匆豢? ></p><p style=如有收獲,點個在看,誠摯感謝

    免責聲明:本文內容由21ic獲得授權后發(fā)布,版權歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

    本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯(lián)系該專欄作者,如若文章內容侵犯您的權益,請及時聯(lián)系本站刪除。
    換一批
    延伸閱讀

    LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

    關鍵字: 驅動電源

    在工業(yè)自動化蓬勃發(fā)展的當下,工業(yè)電機作為核心動力設備,其驅動電源的性能直接關系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護是驅動電源設計中至關重要的兩個環(huán)節(jié),集成化方案的設計成為提升電機驅動性能的關鍵。

    關鍵字: 工業(yè)電機 驅動電源

    LED 驅動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設備的使用壽命。然而,在實際應用中,LED 驅動電源易損壞的問題卻十分常見,不僅增加了維護成本,還影響了用戶體驗。要解決這一問題,需從設計、生...

    關鍵字: 驅動電源 照明系統(tǒng) 散熱

    根據LED驅動電源的公式,電感內電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

    關鍵字: LED 設計 驅動電源

    電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產業(yè)的重要發(fā)展方向。電動汽車的核心技術之一是電機驅動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機驅動系統(tǒng)中的關鍵元件,其性能直接影響到電動汽車的動力性能和...

    關鍵字: 電動汽車 新能源 驅動電源

    在現代城市建設中,街道及停車場照明作為基礎設施的重要組成部分,其質量和效率直接關系到城市的公共安全、居民生活質量和能源利用效率。隨著科技的進步,高亮度白光發(fā)光二極管(LED)因其獨特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

    關鍵字: 發(fā)光二極管 驅動電源 LED

    LED通用照明設計工程師會遇到許多挑戰(zhàn),如功率密度、功率因數校正(PFC)、空間受限和可靠性等。

    關鍵字: LED 驅動電源 功率因數校正

    在LED照明技術日益普及的今天,LED驅動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

    關鍵字: LED照明技術 電磁干擾 驅動電源

    開關電源具有效率高的特性,而且開關電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機重量也有所下降,所以,現在的LED驅動電源

    關鍵字: LED 驅動電源 開關電源

    LED驅動電源是把電源供應轉換為特定的電壓電流以驅動LED發(fā)光的電壓轉換器,通常情況下:LED驅動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

    關鍵字: LED 隧道燈 驅動電源
    關閉