在函數(shù)中定義的一些基本類(lèi)型的變量和對(duì)象的引用變量都是在函數(shù)的棧內(nèi)存中分配,當(dāng)在一段代碼塊定義一個(gè)變量時(shí),Java 就在棧中為這個(gè)變量分配內(nèi)存空間,當(dāng)超過(guò)變量的作用域后,Java 會(huì)自動(dòng)釋放掉為該變量分配的內(nèi)存空間,該內(nèi)存空間可以立即被另作它用。
堆內(nèi)存用來(lái)存放由 new 創(chuàng)建的對(duì)象和數(shù)組,在堆中分配的內(nèi)存,由 Java 虛擬機(jī)的自動(dòng)垃圾回收器來(lái)管理。在堆中產(chǎn)生了一個(gè)數(shù)組或者對(duì)象之后,還可以在棧中定義一個(gè)特殊的變量,讓棧中的這個(gè)變量的取值等于數(shù)組或?qū)ο笤诙褍?nèi)存中的首地址,棧中的這個(gè)變量就成了數(shù)組或?qū)ο蟮囊米兞浚院缶涂梢栽诔绦蛑惺褂脳V械囊米兞縼?lái)訪(fǎng)問(wèn)堆中的數(shù)組或者對(duì)象,引用變量就相當(dāng)于是為數(shù)組或者對(duì)象起的一個(gè)名稱(chēng)。引用變量是普通的變量,定義時(shí)在棧中分配,引用變量在程序運(yùn)行到其作用域之外后被釋放。而數(shù)組和對(duì)象本身在堆中分配,即使程序運(yùn)行到使用 new 產(chǎn)生數(shù)組或者對(duì)象的語(yǔ)句所在的代碼塊之外,數(shù)組和對(duì)象本身占據(jù)的內(nèi)存不會(huì)被釋放,數(shù)組和對(duì)象在沒(méi)有引用變量指向它的時(shí)候,才變?yōu)槔?,不能在被使用,但仍然占?jù)內(nèi)存空間不放,在隨后的一個(gè)不確定的時(shí)間被垃圾回收器收走(釋放掉)。
這也是 Java 比較占內(nèi)存的原因,實(shí)際上,棧中的變量指向堆內(nèi)存中的變量,這就是 Java 中的指針!
堆和棧都是Java用來(lái)在RAM中存放數(shù)據(jù)的地方。
一、堆
(1)Java的堆是一個(gè)運(yùn)行時(shí)數(shù)據(jù)區(qū),類(lèi)的對(duì)象從堆中分配空間。這些對(duì)象通過(guò)new等指令建立,通過(guò)垃圾回收器來(lái)銷(xiāo)毀。
(2)堆的優(yōu)勢(shì)是可以動(dòng)態(tài)地分配內(nèi)存空間,需要多少內(nèi)存空間不必事先告訴編譯器,因?yàn)樗窃谶\(yùn)行時(shí)動(dòng)態(tài)分配的。但缺點(diǎn)是,由于需要在運(yùn)行時(shí)動(dòng)態(tài)分配內(nèi)存,所以存取速度較慢。
二、棧
(1)棧中主要存放一些基本數(shù)據(jù)類(lèi)型的變量(byte,short,int,long,float,double,boolean,char)和對(duì)象的引用。
(2)棧的優(yōu)勢(shì)是,存取速度比堆快,棧數(shù)據(jù)可以共享。但缺點(diǎn)是,存放在棧中的數(shù)據(jù)占用多少內(nèi)存空間需要在編譯時(shí)確定下來(lái),缺乏靈活性。
三、舉例說(shuō)明棧數(shù)據(jù)可以共享
String 可以用以下兩種方式來(lái)創(chuàng)建:
String str1 = new String("abc");
String str2 = "abc";
第一種使用new來(lái)創(chuàng)建的對(duì)象,它存放在堆中。每調(diào)用一次就創(chuàng)建一個(gè)新的對(duì)象。
第二種是先在棧中創(chuàng)建對(duì)象的引用str2,然后查找棧中有沒(méi)有存放“abc”,如果沒(méi)有,則將“abc”存放進(jìn)棧,并將str2指向“abc”,如果已經(jīng)有“abc”, 則直接將str2指向“abc”。