您好,登錄后才能下訂單哦!
本篇內容主要講解“Java對象的內存分配是怎么保證線程安全的”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“Java對象的內存分配是怎么保證線程安全的”吧!
JVM內存結構,是很重要的知識,相信每一個靜心準備過面試的程序員都可以清楚的把堆、棧、方法區等介紹的比較清楚。
上圖,是一張在作者根據《Java虛擬機規范(Java SE 8)》中描述的JVM運行時內存區域結構畫的。
很多人都知道Java對象是在堆內存中分配空間的(JIT優化除外),也知道內存分配過程中是線程安全的,那么虛擬機到底是如何保證線程安全的呢?本文就來簡單介紹一下。
1.Java對象的內存分配
我們知道,Java是一門面向對象的語言,我們在Java中使用的對象都需要被創建出來,在Java中,創建一個對象的方法有很多種,如使用new、使用反射、使用Clone方法等,但是無論如何,對象在創建過程中,都需要進行內存分配。
拿最常見的new關鍵字舉例,當我們使用new創建對象后代碼開始運行后,虛擬機執行到這條new指令的時候,會先檢查要new的對象對應的類是否已被加載,如果沒有被加載則先進行類加載。
在類加載檢查通過之后,就需要給對象進行內存分配了,分配的內存主要用來存放對象的實例變量。
在進行內存分配時,需要根據對象中的實例變量情況等信息確定需要分配的空間大小,然后從Java堆中劃分出這樣一塊區域(假設沒有JIT優化)。
根據JVM使用的垃圾回收器的類型,因其回收算法不同,會導致堆中內存分配情況不同。如標記-清楚算法回收后的內存中會有大量不連續的內存碎片,在給新的對象分配的時候,就需要通過"空閑列表"來確定一塊空閑區域。(這部分不是本文重點,讀者可以自行學習一下。)
無論那種方式,最終都需要確定出一塊內存區域,用于給新建對象分配內存。我們知道,對象的內存分配過程中,主要是對象的引用指向這個內存區域,然后進行初始化操作。
那么問題就來了:
在并發場景中,如何內存分配過程的線程安全性?如果兩個線程先后把對象引用指向了同一個內存區域,怎么辦。
2.TLAB
一般有兩種解決方案:
1、對分配內存空間的動作做同步處理,采用CAS機制,配合失敗重試的方式保證更新操作的線程安全性。
2、每個線程在Java堆中預先分配一小塊內存,然后再給對象分配內存的時候,直接在自己這塊"私有"內存中分配,當這部分區域用完之后,再分配新的"私有"內存。
方案1在每次分配時都需要進行同步控制,這種是比較低效的。
方案2是HotSpot虛擬機中采用的,這種方案被稱之為TLAB分配,即Thread Local Allocation Buffer。這部分Buffer是從堆中劃分出來的,但是是本地線程獨享的。
這里值得注意的是,我們說TLAB時線程獨享的,但是只是在“分配”這個動作上是線程獨占的,至于在讀取、垃圾回收等動作上都是線程共享的。而且在使用上也沒有什么區別。
另外,TLAB僅作用于新生代的Eden Space,對象被創建的時候首先放到這個區域,但是新生代分配不了內存的大對象會直接進入老年代。因此在編寫Java程序時,通常多個小的對象比大的對象分配起來更加高效。
所以,雖然對象剛開始可能通過TLAB分配內存,存放在Eden區,但是還是會被垃圾回收或者被移到Survivor Space、Old Gen等。
不知道大家有沒有想過,我們使用了TLAB之后,在TLAB上給對象分配內存時線程獨享的了,這就沒有沖突了,但是,TLAB這塊內存自身從堆中劃分出來的過程也可能存在內存安全問題啊。
所以,在對于TLAB的分配過程,還是需要進行同步控制的。但是這種開銷相比于每次為單個對象劃分內存時候對進行同步控制的要低的多。
虛擬機是否使用TLAB是可以選擇的,可以通過設置-XX:+/-UseTLAB參數來指定。
到此,相信大家對“Java對象的內存分配是怎么保證線程安全的”有了更深的了解,不妨來實際操作一番吧!這里是億速云網站,更多相關內容可以進入相關頻道進行查詢,關注我們,繼續學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。