隨筆 - 9  文章 - 21  trackbacks - 0
          <2008年5月>
          27282930123
          45678910
          11121314151617
          18192021222324
          25262728293031
          1234567

          常用鏈接

          留言簿(1)

          隨筆分類(9)

          隨筆檔案(9)

          搜索

          •  

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

          第 5 天的問題

          下面的程序是對(duì)兩個(gè)十六進(jìn)制(hex)字面常量進(jìn)行相加,然后打印出十六進(jìn)制的結(jié)果。這個(gè)程序會(huì)打印出什么呢?

          public class JoyOfHex {
              public static void main(String[] args) {
                  System.out.println(Long.toHexString(0x100000000L + 0xcafebabe));
              }
          }
          

          第 5 天問題的解答

          看起來很明顯,該程序應(yīng)該打印出1cafebabe。畢竟,這確實(shí)就是十六進(jìn)制數(shù)字10000000016與cafebabe16的和。該程序使用的是long型運(yùn)算,它可以支持16位十六進(jìn)制數(shù),因此運(yùn)算溢出是不可能的。

          遺憾的告訴你,結(jié)果是cafebabe,并沒有任何前導(dǎo)的1。這個(gè)輸出表示的是正確結(jié)果的低32位,但是不知何故,第33位丟失了。

          原因

          十進(jìn)制字面常量具有一個(gè)很好的屬性,即所有的十進(jìn)制字面常量都是正的,而十六進(jìn)制或是八進(jìn)制字面常量并不具備這個(gè)屬性。

          要想書寫一個(gè)負(fù)的十進(jìn)制常量,可以使用一元取反操作符(-)連接一個(gè)十進(jìn)制字面常量。以這種方式,你可以用十進(jìn)制來書寫任何int或long型的數(shù)值,不管它是正的還是負(fù)的,并且負(fù)的十進(jìn)制常數(shù)可以很明確地用一個(gè)減號(hào)符號(hào)來標(biāo)識(shí)。

          但是十六進(jìn)制和八進(jìn)制字面常量并不是這么回事,它們可以具有正的以及負(fù)的數(shù)值。如果十六進(jìn)制和八進(jìn)制字面常量的最高位被置位了,那么它們就是負(fù)數(shù)。在這個(gè)程序中,數(shù)字0xcafebabe是一個(gè)int常量,它的最高位被置位了,所以它是一個(gè)負(fù)數(shù)。它等于十進(jìn)制數(shù)值-889275714。

          該程序執(zhí)行的這個(gè)加法是一種"混合類型的計(jì)算(mixed-type computation)。左操作數(shù)是 long 類型的,而右操作數(shù)是 int 類型的。為了執(zhí)行該計(jì)算,Java將int類型的數(shù)值用拓寬原始類型轉(zhuǎn)換提升為一個(gè)long類型,然后對(duì)兩個(gè)long類型數(shù)值相加。因?yàn)閕nt是一個(gè)有符號(hào)的整數(shù)類型,所以這個(gè)轉(zhuǎn)換執(zhí)行的是符合擴(kuò)展,它將負(fù)的int類型的數(shù)值提升為一個(gè)在數(shù)值上相等的long類型數(shù)值。

          解決辦法

          System.out.println(Long.toHexString(0x100000000L + 0xcafebabeL));
          

          這個(gè)加法的右操作數(shù)0xcafebabe被提升為了long類型的數(shù)值0xffffffffcafebabeL。這個(gè)數(shù)值之后被加到了左操作數(shù)0x100000000L上。當(dāng)作為int類型來被審視時(shí),經(jīng)過符號(hào)擴(kuò)展之后的右操作數(shù)的高32位是-1,而左操作數(shù)的高32位是1,將這兩個(gè)數(shù)值相加就得到了0,這也就解釋了為什么在程序輸出中前導(dǎo)1丟失了。

          第 5 天問題的總結(jié)

          這個(gè)題給我們的教訓(xùn)是:混合類型的計(jì)算可能會(huì)產(chǎn)生混淆,尤其是十六進(jìn)制和八進(jìn)制字面常量無(wú)需顯式的減號(hào)符號(hào)就可以表示負(fù)的數(shù)值。為了避免這種窘境,通常最好是避免混合類型的計(jì)算

          對(duì)于語(yǔ)言的設(shè)計(jì)者們來說,應(yīng)該考慮支持無(wú)符號(hào)的整數(shù)類型,從而根除符號(hào)擴(kuò)展的可能性??赡軙?huì)有這樣的爭(zhēng)辯:負(fù)的十六進(jìn)制和八進(jìn)制字面常量應(yīng)該被禁用,但是這可能會(huì)挫傷程序員,他們經(jīng)常使用十六進(jìn)制字面常量來表示那些符號(hào)沒有任何重要含義的數(shù)值。


          今天的問題

          轉(zhuǎn)型被用來將一個(gè)數(shù)值從一種類型轉(zhuǎn)換到另一種類型。下面的程序連續(xù)使用了三個(gè)轉(zhuǎn)型。那么它到底會(huì)打印出什么呢?

          public class Multicast {
              public static void main(String[] args) {
                  System.out.println((int) (char) (byte) -1);
              }
          }
          
          
          posted on 2008-05-17 17:52 李四飛刀 閱讀(1586) 評(píng)論(2)  編輯  收藏 所屬分類: 每日一題

          FeedBack:
          # re: 第 6 天: 解答 -- 16進(jìn)制的趣事, 問題 -- 轉(zhuǎn)型 2008-05-19 12:55 草兒
          你是不是建議我們都去買一本 《Java 解惑》?   回復(fù)  更多評(píng)論
            
          # re: 第 6 天: 解答 -- 16進(jìn)制的趣事, 問題 -- 轉(zhuǎn)型 2008-05-21 16:29 d
          很無(wú)聊!?。。。。?!,如果你寫過c++,你說的都是入門基礎(chǔ)!  回復(fù)  更多評(píng)論
            
          主站蜘蛛池模板: 昌吉市| 伊宁市| 鱼台县| 达拉特旗| 郸城县| 江达县| 淮南市| 天台县| 遵义县| 环江| 丰台区| 永济市| 长丰县| 西城区| 连山| 台北县| 临沂市| 大埔县| 德惠市| 武夷山市| 平顺县| 梁河县| 启东市| 晋州市| 密云县| 淮阳县| 南投县| 饶平县| 靖州| 于田县| 日照市| 白玉县| 新乡县| 泽库县| 金坛市| 临沂市| 汨罗市| 白河县| 景宁| 定日县| 维西|