1. 程式人生 > >java核心(五):堆內存、棧內存、直接內存

java核心(五):堆內存、棧內存、直接內存

fifo 創建 inpu 申請 棧內存 先進先出 span size -a

一、什麽是堆內存、棧內存?   Java把內存劃分成兩種:一種是堆內存,一種是棧內存。   堆:主要用於存儲實例化的對象,數組。由JVM動態分配內存空間。一個JVM只有一個堆內存,線程是可以共享數據的。   棧:主要用於存儲局部變量和對象的引用變量,每個線程都會有一個獨立的棧空間,所以線程之間是不共享數據的。   在函數中定義的一些基本類型的變量和對象的引用變量都在函數的棧內存中分配。 當在一段代碼塊定義一個變量時,Java就在棧中為這個變量分配內存空間,當超過變量的作用域後,Java會自動釋放掉為該變量所分配的內存空間,該內存空間可以立即被另作他用。   堆內存用來存放由 new 創建的對象和數組,在堆中分配的內存,由 Java 虛擬機的自動垃圾回收器來管理。
在堆中產生了一個數組或者對象之後,還可以在棧中定義一個特殊的變量(變量的引用),讓棧中的這個變量的取值等於數組或對象在堆內存中的首地址,棧中的這個變量就成了數組或對象的引用變量,以後就可以在程序中使用棧中的引用變量來訪問堆中的數組或者對象,引用變量就相當於是為數組或者對象起的一個名稱。引用變量是普通的變量,定義時在棧中分配,引用變量在程序運行到其作用域之外後被釋放。而數組和對象本身在堆中分配,即使程序運行到使用 new 產生數組或者對象的語句所在的代碼塊之外,數組和對象本身占據的內存不會被釋放,數組和對象在沒有引用變量指向它的時候,才變為垃圾,不能在被使用,但仍然占據內存空間不放,在隨後的一個不確定的時間被垃圾回收器收走(釋放掉)。 二、堆、棧的特點
  • 堆:FIFO隊列優先,先進先出。jvm只有一個堆區被所有線程所共享!堆存放在二級緩存中,調用對象的速度相對慢一些,生命周期由虛擬機的垃圾回收機制定。
  • 棧:FILO先進後出,暫存數據的地方。每個線程都包含一個棧區!棧存放在一級緩存中,存取速度較快,“棧是限定僅在表頭進行插入和刪除操作的線性表”。

三、堆、棧的優缺點

  • 堆的優點-可以動態的分配內存大小,生命周期不確定。缺點-速度略慢
  • 棧的優點-速度快,缺點-存在棧中的數據大小和生命周期必須是明確的,缺少靈活性。

技術分享圖片

四、java中變量在內存中的分配
  • 類變量(static修飾的變量):在程序加載時,系統就為它在堆中開辟了內存堆中的內存地址存放於棧以便於高速訪問。靜態變量的生命周期--一直持續到整個"系統"關閉
  • 實例變量:當你使用java關鍵字new的時候,系統在堆中開辟並不一定是連續的空間分配給變量(比如說類實例),然後根據零散的堆內存地址,通過哈希算法換算為一長串數字以表征這個變量在堆中的"物理位置"。 實例變量的生命周期--當實例變量的引用丟失後,將被GC(垃圾回收器)列入可回收“名單”中,但並不是馬上就釋放堆中內存
  • 局部變量:局部變量,由聲明在某方法,或某代碼段裏(比如for循環),執行到它的時候在棧中開辟內存,當局部變量一但脫離作用域,內存立即釋放

五、直接內存

  直接內存並不是虛擬機運行時數據區的一部分,也不是Java 虛擬機規範中農定義的內存區域。在JDK1.4 中新加入了NIO(New Input/Output)類,引入了一種基於通道(Channel)與緩沖區(Buffer)的I/O 方式,它可以使用native 函數庫直接分配堆外內存,然後通脫一個存儲在Java堆中的DirectByteBuffer 對象作為這塊內存的引用進行操作。這樣能在一些場景中顯著提高性能,因為避免了在Java堆和Native堆中來回復制數據。

  本機直接內存的分配不會受到Java 堆大小的限制,受到本機總內存大小限制

  配置虛擬機參數時,不要忽略直接內存 防止出現OutOfMemoryError異常

六、直接內存(堆外內存)與堆內存比較

  • 直接內存申請空間耗費更高的性能,當頻繁申請到一定量時尤為明顯
  • 直接內存IO讀寫的性能要優於普通的堆內存,在多次讀寫操作的情況下差異明顯

七、推薦閱讀

  • Java堆和棧的區別,JVM堆和棧的介紹

  • java虛擬機(JVM)堆、棧、方法區的詳細詳解和區別介紹

  • JVM內存結構——堆、棧、方法區、直接內存、堆和棧區別

java核心(五):堆內存、棧內存、直接內存