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