java核心(五):堆內存、棧內存;String賦值時,內存變化

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

3、堆、棧的優缺點html

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

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

5、直接內存java

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

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

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

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

  • 直接內存申請空間耗費更高的性能,當頻繁申請到必定量時尤其明顯
  • 直接內存IO讀寫的性能要優於普通的堆內存,在屢次讀寫操做的狀況下差別明顯

7、String賦值時,內存變化jvm

 

8、更多介紹函數

相關文章
相關標籤/搜索