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

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

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

    我的漫漫程序之旅

    專注于JavaWeb開發
    隨筆 - 39, 文章 - 310, 評論 - 411, 引用 - 0
    數據加載中……

    Java編譯器對于String常量表達式的優化

     

    首先把問題擺出來,先看這個代碼

    String a = "ab";
                String b = "a" + "b";
                System.out.println((a == b));

    打印結果會是什么?類似這樣的問題,有人考過我,我也拿來考過別人(蠻好玩的,大家也可以拿來問人玩),一般答案會是以下幾種:

    1.true

    "a" + "b" 的結果就是"ab",這樣a,b都是"ab"了,內容一樣所以"相等",結果true

    一般java新人如是答。

    2.false

    "a" + "a"會生成新的對象"aa",但是這個對象和String a = "ab";不同,(a == b)是比較對象引用,因此不相等,結果false

    對java的String有一定了解的通常這樣回答。

    3.true

    String a = "ab";創建了新的對象"ab"; 再執行String b = "a" + "b";結果b="ab",這里沒有創建新的對象,而是從JVM字符串常量池中獲取之前已經存在的"ab"對象。因此a,b具有對同一個string對象的引用,兩個引用相等,結果true.

    能回答出這個答案的,基本已經是高手了,對java中的string機制比較了解。

    很遺憾,這個答案,是不夠準確的。或者說,根本沒有運行時計算b = "a" + "b";這個操作.實際上運行時只有String b = "ab";

    3的觀點適合解釋以下情況:

    String a = "ab";
                String b = "ab";
                System.out.println((a == b));

    如果String b = "a" + "b";是在運行期執行,則3的觀點是無法解釋的。運行期的兩個string相加,會產生新的對象的。(本文后面對此有解釋)

    4.true

    下面是我的回答:編譯優化+ 3的處理方式 = 最后的true

    String b = "a" + "b";編譯器將這個"a" + "b"作為常量表達式,在編譯時進行優化,直接取結果"ab",這樣這個問題退化

    String a = "ab";
                String b = "ab";
                System.out.println((a == b));

    然后根據3的解釋,得到結果true

    這里有一個疑問就是String不是基本類型,像

    int secondsOfDay = 24 * 60 * 60;

    這樣的表達式是常量表達式,編譯器在編譯時直接計算容易理解,而"a" + "b" 這樣的表達式,string是對象不是基本類型,編譯器會把它當成常量表達式來優化嗎?

    下面簡單證明我的推斷,首先編譯這個類:

    public class Test {
                private String a = "aa";
                }

    復制class文件備用,然后修改為

    public class Test {
                private String a = "a" + "a";
                }

    再次編譯,用ue之類的文本編輯器打開,察看二進制內容,可以發現,兩個class文件完全一致,連一個字節都不差.

    ok,真相大白了.根本不存在運行期的處理String b = "a" + "b";這樣的代碼的問題,編譯時就直接優化掉了。

    下面進一步探討,什么樣的string + 表達式會被編譯器當成常量表達式?

    String b = "a" + "b";

    這個String + String被正式是ok的,那么string + 基本類型呢?

    String a = "a1";
                String b = "a" + 1;
                System.out.println((a == b)); //result = true
                String a = "atrue";
                String b = "a" + true;
                System.out.println((a == b)); //result = true
                String a = "a3.4";
                String b = "a" + 3.4;
                System.out.println((a == b)); //result = true

    可見編譯器對string + 基本類型是當成常量表達式直接求值來優化的。

    再注意看這里的string都是"**"這樣的,我們換成變量來試試:

    String a = "ab";
                String bb = "b";
                String b = "a" + bb;
                System.out.println((a == b)); //result = false

    這個好理解,"a" + bb中的bb是變量,不能進行優化。這里很很好的解釋了為什么3的觀點不正確,如果String+String的操作是在運行時進行的,則會產生新的對象,而不是直接從jvm的string池中獲取。

    再修改一下,把bb作為常量變量:

    String a = "ab";
                final String bb = "b";
                String b = "a" + bb;
                System.out.println((a == b)); //result = true

    竟然又是true,編譯器的優化好厲害啊,呵呵,考慮下面這種情況:

    String a = "ab";
                final String bb = getBB();
                String b = "a" + bb;
                System.out.println((a == b)); //result = false
                private static String getBB() {
                return "b";
                }

    看來java(包括編譯器和jvm)對string的優化,真的是到了極點了,string這個所謂的"對象",完全不可以看成一般的對象,java對string的處理近乎于基本類型,最大限度的優化了幾乎能優化的地方。

    另外感嘆一下,string的+號處理,算是java語言里面唯一的一個"運算符重載"(接觸過c++的人對這個不會陌生)吧?



    posted on 2007-12-09 16:07 々上善若水々 閱讀(864) 評論(0)  編輯  收藏 所屬分類: J2SE

    主站蜘蛛池模板: 亚洲国产精品一区| 久久久综合亚洲色一区二区三区 | 亚洲精品无码不卡在线播放| 亚洲香蕉免费有线视频| 亚洲第一成年网站大全亚洲| 最近新韩国日本免费观看| 91亚洲导航深夜福利| 国产2021精品视频免费播放| 亚洲成人一级电影| 毛片大全免费观看| 春暖花开亚洲性无区一区二区| 国产午夜鲁丝片AV无码免费 | 久久亚洲国产午夜精品理论片| 香蕉视频在线免费看| 亚洲AV无码不卡在线播放| 最近中文字幕大全中文字幕免费 | 亚洲av午夜成人片精品电影| 一级毛片免费全部播放| 亚洲国产精品一区二区第一页| 99久久久国产精品免费蜜臀 | 国产精品亚洲色婷婷99久久精品| 亚洲国产精品第一区二区三区| 香蕉免费在线视频| 亚洲伊人精品综合在合线| 暖暖日本免费在线视频 | 亚洲黄色在线观看| 夜夜嘿视频免费看| 国产区在线免费观看| 亚洲三级中文字幕| 亚洲伊人久久综合中文成人网| 久久国产乱子伦免费精品| 亚洲精品蜜夜内射| 亚洲色欲一区二区三区在线观看| 国产电影午夜成年免费视频| 亚洲av无码一区二区三区天堂| 亚洲中久无码永久在线观看同| 性短视频在线观看免费不卡流畅| 色五月五月丁香亚洲综合网| 久久精品国产亚洲| 国产精品免费电影| 久久久久久国产精品免费无码 |