<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    ZT文萃

    本博不原創,轉帖自己感興趣那些事人物,什么入眼貼什么,隨心所欲。
    posts - 93, comments - 5, trackbacks - 0, articles - 0
      BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

    轉帖:再解Java中的String

    Posted on 2014-05-04 06:10 ZT文萃 閱讀(216) 評論(0)  編輯  收藏 所屬分類: 中間件

    轉帖:

    http://www.cnblogs.com/chenssy/p/3695271.html
    版權歸作者所有。


         今天朋友問我String的內容是真的不可變嗎?我肯定告訴他是的?因為在我的主觀意識里String就是一個不可變的對象。于是他給我發了這段程序:

    復制代碼
    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的值肯定是可以改變的啦(我們應該始終相信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);     } }
    復制代碼

          乍看這程序是異常的簡單,無非就是賦值、改值、輸出嘛!可能你現在就會毫不猶豫的說太簡單了結果就是……。但是!!你的毫不猶豫會害死你,而且你的結果很可能錯誤。那么運行結果是什么呢?

    復制代碼
    --------------修改前值------------------- a = chenssy b = chenssy c = chenssy --------------修改后值------------------- a = chen_sy b = chen_sy chen_sy c = chen_ssy
    復制代碼

          修改前值很容易理解,但是修改后值呢?是不是有點兒不理解呢?你可能會問:為什么System.out.println("chenssy");的結果會是chen_ssy,System.out.println("c = " + c);也是chen_ssy呢?

          要明白這個其實也比較簡單,掌握一個知識點:字符串常量池。

          我們知道字符串的分配和其他對象分配一樣,是需要消耗高昂的時間和空間的,而且字符串我們使用的非常多。JVM為了提高性能和減少內存的開銷,在實例化字 符串的時候進行了一些優化:使用字符串常量池。每當我們創建字符串常量時,JVM會首先檢查字符串常量池,如果該字符串已經存在常量池中,那么就直接返回 常量池中的實例引用。如果字符串不存在常量池中,就會實例化該字符串并且將其放到常量池中。由于String字符串的不可變性我們可以十分肯定常量池中一定不存在兩個相同的字符串(這點對理解上面至關重要)。

          我們再來理解上面的程序。

          String a = "chenssy";

          String b = "chenssy";

          a、b和字面上的chenssy都是指向JVM字符串常量池中的”chenssy”對象,他們指向同一個對象。

          String c = new String("chenssy");

          new關鍵字一定會產生一個對象chenssy(注意這個chenssy和上面的chenssy不同),同時這個對象是存儲在堆中。所以上面應該產生了兩 個對象:保存在棧中的c和保存堆中chenssy。但是在Java中根本就不存在兩個完全一模一樣的字符串對象。故堆中的chenssy應該是引用字符串 常量池中chenssy。所以c、chenssy、池chenssy的關系應該是:c--->chenssy--->池chenssy。整個 關系如下:

          201404271001

          通過上面的圖我們可以非常清晰的認識他們之間的關系。所以我們修改內存中的值,他變化的是所有。

          總結:雖然a、 b、c、chenssy是不同的對象,但是從String的內部結構我們是可以理解上面的。String c = new String("chenssy");雖然c的內容是創建在堆中,但是他的內部value還是指向JVM常量池的chenssy的value,它構造 chenssy時所用的參數依然是chenssy字符串常量。

          為了讓各位充分理解常量池,特意準備了如下一個簡單的題目:

    String a = "chen"; String b = a + new String("ssy");
             創建了幾個String對象??
    主站蜘蛛池模板: 亚洲av日韩av不卡在线观看| 亚洲国产精品成人午夜在线观看| 91精品成人免费国产片| 亚洲性无码一区二区三区| 久久乐国产精品亚洲综合| 69成人免费视频| www.xxxx.com日本免费| 亚洲日本在线免费观看| 亚洲国产人成精品| 无限动漫网在线观看免费| 国产精品极品美女自在线观看免费 | 亚洲中文字幕久久精品无码A| 亚洲国产V高清在线观看| 最刺激黄a大片免费网站| 老司机午夜性生免费福利| 亚洲色图古典武侠| 亚洲七七久久精品中文国产| 无码永久免费AV网站| 中国极品美軳免费观看| 亚洲天然素人无码专区| 亚洲成年人在线观看| 免费少妇a级毛片| 永久免费av无码不卡在线观看| 你懂的网址免费国产| 国产成人va亚洲电影| 亚洲一区二区三区免费在线观看 | 午夜dj免费在线观看| 久久免费国产视频| 一级大黄美女免费播放| 亚洲乱码无人区卡1卡2卡3| 亚洲高清免费在线观看| 亚洲人成网7777777国产| 麻豆国产精品入口免费观看| www视频免费看| 久久国产乱子伦精品免费不卡| 一道本在线免费视频| 狠狠入ady亚洲精品| 亚洲熟妇AV一区二区三区宅男| 亚洲最大免费视频网| 99ri精品国产亚洲| 亚洲男人第一av网站|