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