一.byte和int相互轉換的方法 java程式或Android程式的socket數據傳輸,都是通過byte數組,但是int類型是4個byte組成的,如何把一個整形int轉換成byte數組,同時如何把一個長度為4的byte數組轉換為int類型。 /** * int到byte[] * @param i ...
一.byte和int相互轉換的方法
java程式或Android程式的socket數據傳輸,都是通過byte數組,但是int類型是4個byte組成的,如何把一個整形int轉換成byte數組,同時如何把一個長度為4的byte數組轉換為int類型。
/** * int到byte[] * @param i * @return */ public static byte[] intToByteArray(int i) { byte[] result = new byte[4]; // 由高位到低位 result[0] = (byte) ((i >> 24) & 0xFF); result[1] = (byte) ((i >> 16) & 0xFF); result[2] = (byte) ((i >> 8) & 0xFF); result[3] = (byte) (i & 0xFF); return result; } /** * byte[]轉int * @param bytes * @return */ public static int byteArrayToInt(byte[] bytes) { int value = 0; // 由高位到低位 for (int i = 0; i < 4; i++) { int shift = (4 - 1 - i) * 8; value += (bytes[i] & 0x000000FF) << shift;// 往高位游 } return value; } //測試數據 public static void main(String[] args) { byte[] b = intToByteArray(128); System.out.println(Arrays.toString(b));//列印byte的每一個位元組 int i = byteArrayToInt(b); System.out.println(i); //列印byte轉變為Int後的數據 }
二.byte和String相互轉換
上面只說到byte和int的相互轉換,其實呢byte和String也是可以相互轉換的,畢竟Socket通信也是需要傳遞字元串的。
其他的方面,char和long、float這些數據都是可以轉換為byte的,但是實際應用場合比較少。
//String 和byte相互轉換的示例 String string = "hello 世界小姐"; byte[] bytes = string.getBytes();//獲得byte數組 System.out.println("bytes-->" + Arrays.toString(bytes));//列印byte數組 System.out.println("string-->" + new String(bytes));//獲得byte數組轉換來的String數據,並列印
上面第一第二都是byte數組和其他數據相互轉換,其實也可以通過流的來獲取位元組,傳遞過去後再把位元組放到流裡面去,解封出來,但是那樣還是太麻煩的!不建議做。
三.byte占用位元組大小詳解
在Java中一共有8種基本數據類型,其中有4種整型,2種浮點類型,1種用於表示Unicode編碼的字元單元的字元類型和1種用於表示真值的boolean類型。(一個位元組等於8個bit)
1.整型
類型 存儲需求 bit 取值範圍 備註
int 4位元組 4*8 很大
short 2位元組 2*8 -32768~32767
long 8位元組 8*8 非常大
byte 1位元組 1*8 -128~127
2.浮點型
類型 存儲需求 bit 取值範圍 備註
float 4位元組 4*8 很大 float類型的數值有一個尾碼F(例如:3.14F)
double 8位元組 8*8 非常大 沒有尾碼F的浮點數值(如3.14)預設為double類型
3.char類型
類型 存儲需求 bit 取值範圍 備註
char 2位元組 2*8 -32768~32767
4.boolean類型
類型 存儲需求 bit 取值範圍 備註
boolean 1位元組 1*8 false、true
四.String與byte[]位元組數組中文轉換亂碼問題
在Java中,String.getBytes(String decode)方法會根據指定的decode編碼返回某字元串在該編碼下的byte數組表示,如
byte[] b_gbk = “中”.getBytes(“GBK”);
byte[] b_utf8 = “中”.getBytes(“UTF-8”);
byte[] b_iso88591 = “中”.getBytes(“ISO8859-1”);
byte[] b_unicode = “中”.getBytes(“unicode”);
將分別返回“中”這個漢字在GBK、UTF-8和ISO8859-1編碼下的byte數組表示,
此時b_gbk的長度為2,b_utf8的長度為3,b_iso88591的長度為1,
b_unicode 的長度為4(系統的的unicode採用的是big-endian就是前面是兩個位元組來表示這個的,unicode採用的都是兩個位元組編碼,所以後面是4個位元組 )。
而與getBytes相對的,可以通過new String(byte[], decode)的方式來還原這個“中”字時,這個new String(byte[], decode)實際是使用decode指定的編碼來將byte[]解析成字元串。
String s_gbk = new String(b_gbk,”GBK”);
String s_utf8 = new String(b_utf8,”UTF-8”);
String s_iso88591 = new String(b_iso88591,”ISO8859-1”);
通過列印s_gbk、s_utf8和s_iso88591,會發現,s_gbk和s_utf8都是“中”,而只有s_iso88591是一個不認識的字元,為什麼使用ISO8859-1編碼再組合之後,
無法還原“中”字呢,其實原因很簡單,因為ISO8859-1編碼的編碼表中,
根本就沒有包含漢字字元,當然也就無法通過”中”.getBytes(“ISO8859-1”);來得到正確的“中”字在ISO8859-1中的編碼值了,所以再通過new String()來還原就無從談起了。
因此,通過String.getBytes(String decode)方法來得到byte[]時,一定要確定decode的編碼表中確實存在String表示的碼值,這樣得到的byte[]數組才能正確被還原。
有時候,為了讓中文字元適應某些特殊要求(如http header頭要求其內容必須為iso8859-1 編碼),可能會通過將中文字元按照位元組方式來編碼的情況,
如 String s_iso88591 = new String(“中”.getBytes(“UTF-8”),”ISO8859-1”),這樣得到的s_iso8859-1字元串實際是三個在ISO8859-1中的字元,
在將這些字元傳遞到目的地後,目的地程式再通過相反的方式String s_utf8 = new String(s_iso88591.getBytes(“ISO8859-1”),”UTF-8”)來得到正確的中文漢字“中”。
這樣就既保證了遵守協議規定、也支持中文。
String的getBytes()方法是得到一個字串的位元組數組,這是眾所周知的。但特別要註意的是,本方法將返回該操作系統預設的編碼格式的位元組數組。
如果你在使用這個方法時不考慮到這一點,你會發現在一個平臺上運行. 良好的系統,放到另外一臺機器後會產生意想不到的問題。
對也String來說,一個英文字元固定占1個位元組,而中文字元占2個(GBK編碼)或3個(UTF-8編碼)位元組。
五.byte位元組其他的知識
兩個位元組能不能表示一個int呢?
其實也是可以的,就是有一些坑會在裡面!
代碼:
int a = 100; byte b1 = (byte) ((a >> 8) & 0xFF); //高8位 byte b2 = (byte) (a & 0xFF); //低8位 System.out.println(a+"-->"+(b1)+(b2) ); int value = 0; value += (b1 & 0xFF) << 8;// 往高位游 value += (b2 & 0xFF);// System.out.println("value=" + value);
如果a的數值在0到32767之內都是正常的。
正常情況是沒有問題的,像一般字元串的長度用兩個位元組的byte和int相互轉換來表示長度,但是如果是負數就不可以用兩個位元組byte來和int相互轉換了。
因為如果是負數的話符號位就是1了,左移和右移有很大想差別。
像會出現負數的情況還是要使用四個byte來和int做相互轉換傳遞數據。
線上測試一下
如果想簡單驗證也可以使用線上java編譯工具,測試一下:
http://www.runoob.com/try/runcode.php?filename=HelloWorld&type=java
把上面的代碼複製進去就可以了,這個線上的便宜工具只能識別一些簡單的類,像位元組流那些還是不能識別的。
共勉:別忘了你是第一名