java - new + 類名,一定需要申明一個對象嗎?
問題描述
public class CodeBlock02{ { System.out.println('第一代碼塊');}public CodeBlock02() {System.out.println('構(gòu)造方法');}{ System.out.println('第二構(gòu)造塊'); } public static void main(String[] args) { new CodeBlock02(); new CodeBlock02(); new CodeBlock02(); }}
在這里, new CodeBlock02(); 或者換成 CodeBlock02 code = new CodeBlock02();他們是一樣的嗎!
問題解答
回答1:先明確幾個概念,java代碼是跑在jvm中的,而jvm的內(nèi)存區(qū)域劃分為這么幾個模塊:
程序計數(shù)器(Program Counter Register):程序計數(shù)器是一個比較小的內(nèi)存區(qū)域,用于指示當(dāng)前線程所執(zhí)行的字節(jié)碼執(zhí)行到了第幾行,可以理解為是當(dāng)前線程的行號指示器。字節(jié)碼解釋器在工作時,會通過改變這個計數(shù)器的值來取下一條語句指令。
虛擬機棧(JVM Stack):一個線程的每個方法在執(zhí)行的同時,都會創(chuàng)建一個棧幀(Statck Frame),棧幀中存儲的有局部變量表、操作站、動態(tài)鏈接、方法出口等,當(dāng)方法被調(diào)用時,棧幀在JVM棧中入棧,當(dāng)方法執(zhí)行完成時,棧幀出棧。
本地方法棧(Native Method Statck):本地方法棧在作用,運行機制,異常類型等方面都與虛擬機棧相同,唯一的區(qū)別是:虛擬機棧是執(zhí)行Java方法的,而本地方法棧是用來執(zhí)行native方法的,在很多虛擬機中(如Sun的JDK默認的HotSpot虛擬機),會將本地方法棧與虛擬機棧放在一起使用。
堆區(qū)(Heap):堆區(qū)是理解Java GC機制最重要的區(qū)域,沒有之一。在JVM所管理的內(nèi)存中,堆區(qū)是最大的一塊,堆區(qū)也是Java GC機制所管理的主要內(nèi)存區(qū)域,堆區(qū)由所有線程共享,在虛擬機啟動時創(chuàng)建。堆區(qū)的存在是為了存儲對象實例,原則上講,所有的對象都在堆區(qū)上分配內(nèi)存(不過現(xiàn)代技術(shù)里,也不是這么絕對的,也有棧上直接分配的)。
方法區(qū)(Method Area):(也被稱為永久代),方法區(qū)是各個線程共享的區(qū)域,用于存儲已經(jīng)被虛擬機加載的類信息(即加載類時需要加載的信息,包括版本、field、方法、接口等信息)、final常量、靜態(tài)變量、編譯器即時編譯的代碼等。
直接內(nèi)存(Direct Memory):直接內(nèi)存并不是JVM管理的內(nèi)存,可以這樣理解,直接內(nèi)存,就是JVM以外的機器內(nèi)存,比如,你有4G的內(nèi)存,JVM占用了1G,則其余的3G就是直接內(nèi)存,JDK中有一種基于通道(Channel)和緩沖區(qū)(Buffer)的內(nèi)存分配方式,將由C語言實現(xiàn)的native函數(shù)庫分配在直接內(nèi)存中,用存儲在JVM堆中的DirectByteBuffer來引用。由于直接內(nèi)存收到本機器內(nèi)存的限制,所以也可能出現(xiàn)OutOfMemoryError的異常。
明白這幾個基本概念以后再來看看題主疑惑的地方。其實題主疑惑的是在java中,對象的引用是個什么東西,它們和對象的實例化過程有什么關(guān)系。
別急我們先來分析下java中一個引用是怎么實現(xiàn)的:
一個Java的引用訪問涉及到3個內(nèi)存區(qū)域:JVM棧,堆,方法區(qū)。
以最簡單的本地變量引用:Object obj = new Object()為例:
Object obj表示一個本地引用,存儲在JVM棧的本地變量表中,表示一個reference類型數(shù)據(jù);
new Object()作為實例對象數(shù)據(jù)存儲在堆中;
堆中還記錄了Object類的類型信息(接口、方法、field、對象類型等)的地址,這些地址所執(zhí)行的數(shù)據(jù)存儲在方法區(qū)中;
具體的實現(xiàn)方式有很多種,句柄是其中一種,關(guān)系如圖所示。
看到這里應(yīng)該就明白了。類本身的信息,類實例數(shù)據(jù),以及指向?qū)ο蟮囊眯畔⒎謩e放在 java 的方法區(qū)和棧區(qū)以及堆區(qū)。
在題主的例子中:
CodeBlock02 code = new CodeBlock02();
code 就是存放在本地變量表的一個引用,它指向堆中的對象實例數(shù)據(jù)。而這個對象實例數(shù)據(jù),就是通過new CodeBlock02() 取到的。
再具體一點:
1. 你寫的 CodeBlock02.java 文件存放了 CodeBlock02 類的定義,當(dāng) jvm 的類加載器加載這個java文件的時候,將其中的類型定義語句存放在了 jvm 的方法區(qū)中。2. 但是這個時候并沒有在堆中生成這個對象的實例,也就是說,這個時候因為沒有對象,你并不能調(diào)用 CodeBlock02 類的非靜態(tài)方法。3. 什么時候獲取的對象呢?就是在用 new 關(guān)鍵字執(zhí)行了本類的構(gòu)造方法以后 new CodeBlock02() 從這時候開始通過 new 關(guān)鍵字和類的構(gòu)造器, jvm 在虛擬機的堆區(qū)創(chuàng)建了一個 CodeBlock02 類的實例,并返回這個實例的引用,同時你也可以通過這個引用調(diào)用它的非靜態(tài)方法了。
綜上所述,code 就是你用來接收 new 出的實例的的“遙控器”,它指向這個對象在堆區(qū)的具體位置。
回答2:你需要理解 java 的引用
CodeBlock02 code = new CodeBlock02();
左邊這個叫做 CodeBlock02 類型的變量。
右邊這個叫做 CodeBlock02 類型的對象。
你也可以讓這個變量依次指向兩個類型相同的不同對象。
CodeBlock02 code;CodeBlock02 code1 = new CodeBlock02();CodeBlock02 code2 = new CodeBlock02();code = code1;//code.doSomething(); 相當(dāng)于 code1.doSomething();code = code2;//code.doSomething(); 相當(dāng)于 code2.doSomething();
你甚至可以讓這個類型的變量指向這個類型的子類的對象:
MyClass m = new SubMyClass(); //SubMyClass 繼承于 MyClass
還可以這樣直接在 new 出來的對象上調(diào)用方法:
new CodeBlock02().doSomething();回答3:
兩個都是聲明對象 樓主問的應(yīng)該是賦值
如果后面不對這個值繼續(xù)操作的話 賦不賦值都是一樣的
new CodeBlock02() // 聲明了之后不賦值,沒有辦法后續(xù)對這個對象繼續(xù)操作CodeBlock02 code = new CodeBlock02(); // 把聲明的對象賦值給一個變量,可以進行后續(xù)操作
回答4:左邊的是對象的引用變量,右邊的是在內(nèi)存實際分配的對象。
