隨筆-128  評論-55  文章-5  trackbacks-0
          在剖析該問題前請看如下代碼
          public static String bytes2HexString(byte[] b) {
            String ret = "";
            for (int i = 0; i < b.length; i++) {
             String hex = Integer.toHexString(b[ i ] & 0xFF);
             if (hex.length() == 1) {
              hex = '0' + hex;
             }
             ret += hex.toUpperCase();
            }
            return ret;
          }
          上面是將byte[]轉化十六進制的字符串,注意這里b[ i ] & 0xFF將一個byte和 0xFF進行了與運算,然后使用Integer.toHexString取得了十六進制字符串,可以看出
          b[ i ] & 0xFF運算后得出的仍然是個int,那么為何要和 0xFF進行與運算呢?直接 Integer.toHexString(b[ i ]);,將byte強轉為int不行嗎?答案是不行的.

          其原因在于:
          1.byte的大小為8bits而int的大小為32bits
          2.java的二進制采用的是補碼形式

          ==================以下文字貌似有問題===================
          在這里先溫習下計算機基礎理論

          byte是一個字節保存的,有8個位,即8個0、1。
          8位的第一個位是符號位,
          也就是說0000 0001代表的是數字1
          1000 0000代表的就是-1
          所以正數最大位0111 1111,也就是數字127
          負數最大為1111 1111,也就是數字-128

          上面說的是二進制原碼,但是在java中采用的是補碼的形式,下面介紹下什么是補碼

          1、反碼:
                  一個數如果是正,則它的反碼與原碼相同;
                  一個數如果是負,則符號位為1,其余各位是對原碼取反;

          2、補碼:利用溢出,我們可以將減法變成加法
                 對于十進制數,從9得到5可用減法:
                 9-4=5    因為4+6=10,我們可以將6作為4的補數
                 改寫為加法:
                 9+6=15(去掉高位1,也就是減10)得到5.

                 對于十六進制數,從c到5可用減法:
                 c-7=5    因為7+9=16 將9作為7的補數
                 改寫為加法:
                 c+9=15(去掉高位1,也就是減16)得到5.

              在計算機中,如果我們用1個字節表示一個數,一個字節有8位,超過8位就進1,在內存中情況為(100000000),進位1被丟棄。

              ⑴一個數為正,則它的原碼、反碼、補碼相同
              ⑵一個數為負,剛符號位為1,其余各位是對原碼取反,然后整個數加1
             
          - 1的原碼為                10000001
          - 1的反碼為                11111110
                                                             + 1
          - 1的補碼為                11111111

          0的原碼為                 00000000
          0的反碼為                 11111111(正零和負零的反碼相同)
                                                    +1
          0的補碼為               100000000(舍掉打頭的1,正零和負零的補碼相同)

          Integer.toHexString的參數是int,如果不進行&0xff,那么當一個byte會轉換成int時,由于int是32位,而byte只有8位這時會進行補位,
          例如補碼11111111的十進制數為-1轉換為int時變為11111111111111111111111111111111好多1啊,呵呵!即0xffffffff但是這個數是不對的,這種補位就會造成誤差。
          和0xff相與后,高24比特就會被清0了,結果就對了。

          ----
          Java中的一個byte,其范圍是-128~127的,而Integer.toHexString的參數本來是int,如果不進行&0xff,那么當一個byte會轉換成int時,對于負數,會做位擴展,舉例來說,一個byte的-1(即0xff),會被轉換成int的-1(即0xffffffff),那么轉化出的結果就不是我們想要的了。

          而0xff默認是整形,所以,一個byte跟0xff相與會先將那個byte轉化成整形運算,這樣,結果中的高的24個比特就總會被清0,于是結果總是我們想要的。



          Author: orangelizq
          email: orangelizq@163.com

          歡迎大家訪問我的個人網站 萌萌的IT人
          posted on 2008-07-20 20:36 桔子汁 閱讀(56481) 評論(11)  編輯  收藏

          評論:
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2009-04-29 16:55 | mellon
          怎么就造成誤差了呢?如果把0xffffffff來個substring(6)操作,如何?  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2009-07-08 20:56 | baby
          反碼、補碼。。。暈  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2009-08-21 16:06 | 11
          謝謝 ,還好計算機導論沒忘記  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算[未登錄] 2010-06-09 01:00 | 朱亮
          掃盲了  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2011-11-07 16:38 | ff
          11111111111111111111111111111111不還是-1么?會出什么問題呢?  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2011-11-08 11:43 | cuzn
          多謝了,哈哈  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2012-11-09 18:09 |
          java二進制1開頭最大的數才是-1吧
          100.....001是最小的負數  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2013-03-04 13:27 | www
          1000 0000代表的就是-1?
          是這樣嗎?確定-1的byte在java里面不是11111111?  回復  更多評論
            
          # re: [轉]java中byte轉換int時為何與0xff進行與運算 2014-02-09 20:22 | javachow
          lz你是在害人阿~

          “1000 0000代表的就是-1 ”  回復  更多評論
            

          只有注冊用戶登錄后才能發表評論。


          網站導航:
           
          主站蜘蛛池模板: 莆田市| 广河县| 朝阳市| 互助| 饶河县| 玉屏| 汕头市| 宁武县| 商丘市| 仁寿县| 莆田市| 五河县| 广水市| 灵川县| 桂东县| 宁明县| 恩施市| 阜宁县| 河西区| 军事| 夏河县| 石台县| 日照市| 西乌珠穆沁旗| 漯河市| 舒城县| 姚安县| 永安市| 博湖县| 山东省| 深州市| 绥宁县| 承德市| 新民市| 乐陵市| 林州市| 清河县| 肃北| 常德市| 通山县| 于都县|