更新時間:2021-03-29 來源:黑馬程序員 瀏覽量:
應用程序想要完成具體的功能,僅有類是遠遠不夠的,還需要根據(jù)類創(chuàng)建實例對象。在Java程序中,可以使用new關鍵字來創(chuàng)建對象,具體語法格式如下:
類名 對象名稱 = new 類名();例如,創(chuàng)建Person類的實例對象代碼如下:
Person p = new Person();
圖1 內(nèi)存分析
從圖1可以看出,在創(chuàng)建Person對象時,程序會占用兩塊內(nèi)存區(qū)域,分別是棧內(nèi)存和堆內(nèi)存。其中Person類型的變量p被存放在棧內(nèi)存中,它是一個引用,會指向真正的對象;通過new Person()創(chuàng)建的對象則放在堆內(nèi)存中,這才是真正的對象。
小提示:
Java將內(nèi)存分為兩種,即棧內(nèi)存和堆內(nèi)存。其中棧內(nèi)存用于存放基本類型的變量和對象的引用變量(如Person p),堆內(nèi)存用于存放由new創(chuàng)建的對象和數(shù)組。
在創(chuàng)建Person對象后,可以通過對象的引用來訪問對象所有的成員,具體格式如下:
對象引用.對象成員
接下來通過一個案例來學習如何訪問對象的成員,如文件1所示。
文件1 Example02.java
public class Example02 { public static void main(String[] args) { Person p1 = new Person(); // 創(chuàng)建第一個Person類對象 Person p2 = new Person(); // 創(chuàng)建第二個Person類對象 p1.age = 18; // 為age屬性賦值 p1.speak(); // 調(diào)用對象的方法 p2.speak(); } }
運行結果如圖2所示。
圖2 運行結果
文件1中,p1、p2分別引用了Person類的兩個實例對象。從圖2可以看出,p1和p2對象在調(diào)用speak()方法時,打印的age值不同。這是因為p1對象和p2對象是兩個完全獨立的個體,它們分別擁有各自的age屬性,對p1對象的age屬性進行賦值并不會影響到p2對象age屬性的值。程序運行期間p1、p2引用的對象在內(nèi)存中的狀態(tài)如圖3所示。
圖3 P1、P2對象在內(nèi)存中的狀態(tài)
小提示:
在實際情況下,除了可以使用文件3-2中介紹的對象引用來訪問對象成員外,還可以直接使用創(chuàng)建的對象本身來引用對象成員,具體格式如下:
new 類名().對象成員
這種方式是在通過new關鍵字創(chuàng)建實例對象的同時就訪問了對象的某個成員,并且在創(chuàng)建后只能訪問其中某一個成員,而不能像對象引用那樣可以訪問多個對象成員。同時,由于沒有對象引用的存在,在完成某一個對象成員的訪問后,該對象就會變成垃圾對象。所以,在實際開發(fā)中,創(chuàng)建實例對象時多數(shù)會使用對象引用。
在文件1中,通過“p1.age=18”將p1對象的age屬性賦值為18,但并沒有對p2對象的age屬性進行賦值,按理說p2對象的age屬性應該是沒有值的。但從圖2可以看出,p2對象的age屬性也是有值的,其值為0。這是因為在實例化對象時,Java虛擬機會自動為成員變量進行初始化,針對不同類型的成員變量賦予不同的初始值,如表1所示。
表1 成員變量的初始化值
成員變量類型 | 初始值 |
---|---|
byte |
0 |
short | 0 |
int | 0 |
long | 0 |
float | 0.0 |
double | 0.0 |
char | 空字符,'\u0000' |
boolean | false |
引用數(shù)據(jù)類型 | null |
第一段程序代碼:
{ Person p1 = new Person(); ...... }
上面的代碼中,使用變量p1引用了一個Person類型的對象。當這段代碼運行完畢時,變量p1就會超出其作用域而被銷毀,這時Person類型的對象將因為沒有被任何變量所引用而變成垃圾。
第二段程序代碼:
{ Person p2 = new Person(); ...... p2 = null; ...... }
上面的代碼中,使用變量p2引用了一個Person類型的對象,接著將變量p2的值置為null,則表示該變量不指向任何一個對象,被p2所引用的Person對象就會失去引用,成為垃圾對象,過程如圖4所示。
圖4 垃圾對象
猜你喜歡: