Java中的內存分配機制

Java的內存分爲兩種:一種是棧內存,一種是堆內存。java

   在函數中定義的一些基本類型變量和對象的引用都在函數的棧內存中分配。當在一個代碼塊中定義一個變量的時候,java就在棧中爲其分配內存,當超過做用域的時候內存自動釋放。數組

   堆內存用來存放new建立的對象和數組。在堆中分配的內存,由java虛擬機的垃圾回收機器管理。java的堆是運行時數據區,堆的優點是能夠動態的分配內存大小,生存週期也沒必要事先告訴編譯器,可是,因爲是動態分配,存取速度慢。函數

   棧的優點是比堆的存取速度快,僅次於寄存器,棧數據能夠共享,可是棧中的內存大小和生存期必須是固定的,缺少靈活性。測試

棧有一個很重要的特殊性,就是存在棧中的數據能夠共享。假設咱們同時定義:spa

int a = 3;orm

int b = 3;對象

編譯器先處理int a = 3;首先它會在棧中建立一個變量爲a的引用,而後查找棧中是否有3這個值,若是沒找到,就將3存放進來,而後將a指向3。接着處理int b = 3;在建立完b的引用變量後,由於在棧中已經有3這個值,便將b直接指向3。這樣,就出現了ab同時均指向3的狀況。這時,若是再令a=4;那麼編譯器會從新搜索棧中是否有4值,若是沒有,則將4存放進來,並令a指向4;若是已經有了,則直接將a指向這個地址。所以a值的改變不會影響到b的值。要注意這種數據的共享與兩個對象的引用同時指向一個對象的這種共享是不一樣的,由於這種狀況a的修改並不會影響到b, 它是由編譯器完成的,它有利於節省空間。而一個對象引用變量修改了這個對象的內部狀態,會影響到另外一個對象引用變量。內存

String是一個特殊的包裝類數據。能夠用:作用域

第一種:String str = new String("abc");字符串

第二種:String str = "abc";

兩種的形式來建立,第一種是用new()來新建對象的,它會在存放於堆中。每調用一次就會建立一個新的對象。

而第二種是先在棧中建立一個對String類的對象引用變量str,而後查找棧中有沒有存放"abc",若是沒有,則將"abc"存放進棧,並令str指向」abc」,若是已經有」abc」 則直接令str指向「abc」。

比較類裏面的數值是否相等時,用equals()方法;當測試兩個包裝類的引用是否指向同一個對象時,用==,下面用例子說明上面的理論。

String str1 = "abc";

String str2 = "abc";

System.out.println(str1==str2); //true

能夠看出str1str2是指向同一個對象的。

 

String str1 =new String ("abc");

String str2 =new String ("abc");

System.out.println(str1==str2); // false

new的方式是生成不一樣的對象。每一次生成一個。

所以用第二種方式建立多個」abc」字符串,在內存中其實只存在一個對象而已. 這種寫法有利與節省內存空間. 同時它能夠在必定程度上提升程序的運行速度,由於JVM會自動根據棧中數據的實際狀況來決定是否有必要建立新對象。而對於String str = new String("abc");的代碼,則一律在堆中建立新對象,而無論其字符串值是否相等,是否有必要建立新對象,從而加劇了程序的負擔。

另外一方面, 要注意: 咱們在使用諸如String str = "abc";的格式定義類時,老是想固然地認爲,建立了String類的對象str。擔憂陷阱!對象可能並無被建立!而可能只是指向一個先前已經建立的對象。只有經過new()方法才能保證每次都建立一個新的對象。因爲String類的immutable性質,當String變量須要常常變換其值時,應該考慮使用StringBuffer類,以提升程序效率。

相關文章
相關標籤/搜索