再解Java中的String
今天朋友問我String的內(nèi)容是真的不可變嗎?我肯定告訴他是的?因為在我的主觀意識里String就是一個不可變的對象。于是他給我發(fā)了這段程序:
public class StringTest { public static void main(String[] args) throws Exception { String a = "chenssy"; System.out.println("a = " + a); Field a_ = String.class.getDeclaredField("value"); a.setAccessible(true); char[] value=(char[])a.get(a); value[4]='_'; //修改a所指向的值 System.out.println("a = " + a); } } |
看到這個簡單的程序,我笑了,你這不是從底層來修改String的值么?從這里來理解String的值肯定是可以改變的啦(我們應(yīng)該始終相信String的不可變性)!接著他再給我一段程序:
public class StringTest { public static void main(String[] args) throws Exception { String a = "chenssy"; String b = "chenssy"; String c = new String("chenssy"); System.out.println("--------------修改前值-------------------"); System.out.println("a = " + a); System.out.println("b = " + b); System.out.println("c = " + c); //修改String的值 Field a_ = String.class.getDeclaredField("value"); a_.setAccessible(true); char[] value=(char[])a_.get(a); value[4]='_'; //修改a所指向的值 System.out.println("--------------修改后值-------------------"); System.out.println("a = " + a); System.out.println("b = " + b); System.out.println("chenssy"); System.out.println("c = " + c); } } |
字體: 小 中 大 | 上一篇 下一篇 | 打印 | 我要投稿
乍看這程序是異常的簡單,無非就是賦值、改值、輸出嘛!可能你現(xiàn)在就會毫不猶豫的說太簡單了結(jié)果就是……。但是!!你的毫不猶豫會害死你,而且你的結(jié)果很可能錯誤。那么運行結(jié)果是什么呢?
--------------修改前值------------------- a = chenssy b = chenssy c = chenssy --------------修改后值------------------- a = chen_sy b = chen_sy chen_sy c = chen_ssy |
修改前值很容易理解,但是修改后值呢?是不是有點兒不理解呢?你可能會問:為什么System.out.println("chenssy");的結(jié)果會是chen_ssy,System.out.println("c = " + c);也是chen_ssy呢?
要明白這個其實也比較簡單,掌握一個知識點:字符串常量池。
我們知道字符串的分配和其他對象分配一樣,是需要消耗高昂的時間和空間的,而且字符串我們使用的非常多。JVM為了提高性能和減少內(nèi)存的開銷,在實例化字符串的時候進行了一些優(yōu)化:使用字符串常量池。每當(dāng)我們創(chuàng)建字符串常量時,JVM會首先檢查字符串常量池,如果該字符串已經(jīng)存在常量池中,那么就直接返回常量池中的實例引用。如果字符串不存在常量池中,就會實例化該字符串并且將其放到常量池中。由于String字符串的不可變性我們可以十分肯定常量池中一定不存在兩個相同的字符串(這點對理解上面至關(guān)重要)。
我們再來理解上面的程序。
String a = "chenssy";
String b = "chenssy";
a、b和字面上的chenssy都是指向JVM字符串常量池中的”chenssy”對象,他們指向同一個對象。
String c = new String("chenssy");
new關(guān)鍵字一定會產(chǎn)生一個對象chenssy(注意這個chenssy和上面的chenssy不同),同時這個對象是存儲在堆中。所以上面應(yīng)該產(chǎn)生了兩個對象:保存在棧中的c和保存堆中chenssy。但是在Java中根本就不存在兩個完全一模一樣的字符串對象。故堆中的chenssy應(yīng)該是引用字符串常量池中chenssy。所以c、chenssy、池chenssy的關(guān)系應(yīng)該是:c--->chenssy--->池chenssy。整個關(guān)系如下:
通過上面的圖我們可以非常清晰的認(rèn)識他們之間的關(guān)系。所以我們修改內(nèi)存中的值,他變化的是所有。
總結(jié):雖然a、b、c、chenssy是不同的對象,但是從String的內(nèi)部結(jié)構(gòu)我們是可以理解上面的。String c = new String("chenssy");雖然c的內(nèi)容是創(chuàng)建在堆中,但是他的內(nèi)部value還是指向JVM常量池的chenssy的value,它構(gòu)造chenssy時所用的參數(shù)依然是chenssy字符串常量。
posted on 2014-05-08 16:45 順其自然EVO 閱讀(184) 評論(0) 編輯 收藏 所屬分類: 測試學(xué)習(xí)專欄