Java中的byte & 0xff到底有什么作用?
如果寫過通信類的代碼,比如socket編程,應該對這個問題不陌生。
先說結論byte & 0xff 是將byte從(-128 至 127)轉化成 int(轉化后的數值范圍: 0 至 255)。
其實就是1個byte有兩種表示方法,我們既可以用-128 - 127這段范圍來表示一個字節,也可以用 0 - 255這個范圍的數來表示一個字節。
看一個demo用Java中的InetAddress類來獲取我當前的ip
public class InetAddressTest { public static void main(String[] args) throws UnknownHostException {InetAddress localHost = InetAddress.getLocalHost();byte[] address = localHost.getAddress();for (byte b : address) { System.out.print(b + ' ');} }}
輸出結果
-64 -88 2 119
本機ip
好像不太一樣,我們ip地址只用 0 - 255來表示,不會出現負數。
所以再換一種寫法,將取出來的字節 & 0xff
public class InetAddressTest { public static void main(String[] args) throws UnknownHostException {InetAddress localHost = InetAddress.getLocalHost();byte[] address = localHost.getAddress();for (byte b : address) { System.out.print( ( b & 0xff ) + ' ');} }}
結果
192 168 2 119
Process finished with exit code 0
果然就是我們想要的結果
為什么需要轉換因為Java中的byte是有符號的,他的范圍只能是 -128 - 127。
我們在使用tcp等協議的時候,首先要把傳輸的消息轉化成字節流,然后再傳輸,在編程語言中字節流通常用十進制的byte數組來表示。
假如我們就想用 0-255來表示一個字節,不想用負數,該怎么辦呢?
可惜Java中沒有 無符號字節(unsigned byte), 我們只能用 int 來存儲0-255。
而int的范圍是(-2^31 ~ 2^31-1),只用了256個,剩下的空間都被浪費了,得不償失啊。
所以我們存儲的時候、傳輸的時候可以用byte,但是使用的時候就需要做一個轉換了,那為什么0xff就可以得到無符號byte呢。
& 0xff的作用
作為一個十六進制數,0xff在Java中是用什么類型存儲的呢?
應該顯而易見吧,0xff是整型。
假設我現在要轉化 字節 -1
-1的原碼、反碼、補碼分別如下:
原碼 1 0 0 0 0 0 0 1反碼 0 1 1 1 1 1 1 0補碼 0 1 1 1 1 1 1 1
現在和 0xff做運算, ff 就是(1111 11111),而因為他是整型,占4個字節,32為,所以0Xff的前面還有24個0。
用 -1 的補碼進行計算
-1 0 1 1 1 1 1 1 10xff 000000000 000000000 000000000 1 1 1 1 1 1 1 1=000000000 000000000 000000000 0 1 1 1 1 1 1 1=255
其實在Java中,”任何數 & 0Xff等于那個數本身“ 這句話就顯得不那么正確了
”任意整型 & 0xff = 本身“ 是沒有問題的
但是字節 & 0xff 就被拖到了另一個次元,從byte進化成了int。
關于byte[ ] & 0xFF的問題最近在寫有關SHA256加密解密的問題,發現有一段代碼是這樣的,處于好奇理解了一下。
private static String byte2Hex(byte[] bytes){ StringBuffer stringBuffer = new StringBuffer(); String temp = null; for (int i=0;i<bytes.length;i++){ temp = Integer.toHexString(bytes[i] & 0xFF); if (temp.length()==1){ //1得到一位的進行補0操作 stringBuffer.append('0'); } stringBuffer.append(temp); } return stringBuffer.toString(); }
Integer類中toHexString方法的參數是int類型,為什么byte[ ] & 0xFF可以表示int類型呢?
byte[i]是8位二進制,0xFF轉化為8位二進制為11111111,& 之后的結果還是本身啊,這是怎么回事?
我們都知道計算機內的存儲都是利用二進制的補碼進行存儲的。
復習一下,原碼反碼補碼這三個概念對于一個字節的最高位,計算機中是有規定的,正數的最高位為0,負數的最高位為1。
對于正數(00000001)原碼來說,首位表示符號位,反碼 補碼都是本身
對于負數(100000001)原碼來說,反碼是對原碼除了符號位之外作取反運算即(111111110),補碼是對反碼作+1運算即(111111111)
下面寫段代碼測試下
public static void main(String[] args) { byte[] a=new byte[10]; a[0]=-127; System.out.println('a[0]:'+a[0]); int b=a[0] & 0xFF; System.out.println('b:'+b); }
得到的結果為:
a[0]:-127
b:129
現在針對這個結果進行分析:
byte類型的a[0]的值為-127,在計算機中存儲的補碼為:10000001,這個補碼是8位的,而int類型是32位的,所以a[0]作為int類型來輸出的時候jvm給做了個補位便成了 111111111111111111111111 10000001(-127),雖然補碼轉換了,但是這兩個補碼表示的十進制數字是相同的。
為了保證二進制數據的一致性,當byte要轉化為int的時候,高的24位必然會補1,這樣,其二進制補碼其實已經不一致了,如果二進制被當作byte和int來解讀,其10進制的值必然是不同的,因為符號位位置已經發生了變化,而&0xFF可以將高的24位置為0,低8位保持原樣。
int b = a[0]&0xff; a[0]&0xff=1111111111111111111111111 10000001&11111111=000000000000000000000000 10000001 ,這個值就是129
所以最后顯示的b的值為129
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持好吧啦網。
相關文章: