前段時間在工作中遇到一個問題,將整數作為數據傳輸,因為數據包留給我們的字節數也不多,所以需要將int類型轉化為byte[]存放。需要注意的是在java和C#中,byte的取值范圍並不一致。
這里貼出C#代碼,java的實現是一樣的。
int轉字節數組
/// </summary>
/// <param name="number"></param>
/// <returns></returns>
private static byte[] Int32ToBytes(uint number)
{
//int numCopy = number;
byte[] bs = new byte[4];
byte a = (byte)(number >> 24);
byte b = (byte)((number & 0xff0000) >> 16);
byte c = (byte)((number & 0xff00) >> 8);
byte d = (byte)(number & 0xff);
bs[0] = a;
bs[1] = b;
bs[2] = c;
bs[3] = d;
return bs;
}
unit為無符號整型,可表示0-(2^32-1)的整數,byte類型可表示0~255的整數。用二進制表示unit是32位,byte為8位,所以將unit的划分為4個8位分別byte中就可以了。
byte a = (byte)(number >> 24);
將number右移24位,只留下最高8位,放入a中,這里int轉byte還需要一個強轉。
byte b = (byte)((number & 0xff0000) >> 16);
將number的最高8位置0,第二高8位右移到最低8位,放入到b中,c、d也是這樣賦值。
字節數組轉int
private static uint BytesToInt32(byte[] bs)
{
if (bs == null || bs.Length != 4)
{
throw new EncryptionException(ErrorCode.defaultCode, "傳入數組長度不為4");
}
//獲取最高八位
uint num1 = 0;
num1 = (uint)(Convert.ToInt32(num1) ^ (int)bs[0]);
num1 = num1 << 24;
//獲取第二高八位
uint num2 = 0;
num2 = (uint)(Convert.ToInt32(num2) ^ (int)bs[1]);
num2 = num2 << 16;
//獲取第二低八位
uint num3 = 0;
num3 = (uint)(Convert.ToInt32(num3) ^ (int)bs[2]);
num3 = num3 << 8;
//獲取低八位
uint num4 = 0;
num4 = (uint)(Convert.ToInt32(num4) ^ (int)bs[3]);
return num1 ^ num2 ^ num3 ^ num4;
}
byte轉int的思路,先將每個byte位轉為一個32位的整數,再將4個整數做異或,這樣就把4個8位放到一個整數當中。
總結
byte和int類型在計算機底層其實都是用二進制表示的,這樣看其實byte和int只是長度不一樣的一個二進制數而已。需要注意的是在C#中byte和unit都是無符號的整數,所以右移並不存在問題,左邊位補0就行,但是如果有負數的話,符號位是要補1的。