寫在前面
上次在簡書上第一次發(fā)表文章多了30位朋友關(guān)注和收到80多個喜歡讓我受寵若驚。我當時打算以后要多寫一些有價值的文章刘陶,才對得起關(guān)注我的朋友。然而疑苫,愛游戲的我在剛不久前把四公主接回來了~這一個多月都沉迷于游戲世界... 游戲世界也同樣精彩纷责,但現(xiàn)實的世界更精彩~今天就先寫一個簡單的東西,讓我先找回狀態(tài)再膳。
哦!對了喂柒,如果ps4玩家有興趣的話禾嫉,可以私信我或者在下面評論蚊丐,加個好友,有空的話一起玩玩游戲孽椰,雖然我是單機游戲愛好者~
正文
你是否跟我一樣凛篙,在一些需要加密的代碼里看見需要把字符串轉(zhuǎn)換為字節(jié)的場景,例如MD5加密鞋诗。而你每次看到像我以前一樣,自動地略過(反正別人都寫好了全庸,我用就是了)融痛。如果是的話壶笼,不要再逃避了雁刷,下面跟我一起來了解一下字符串與byte之間轉(zhuǎn)換的原理
原理
我們都知道,在Java里byte類型是占用1個字節(jié)责语,即8位的目派,而16進制的字符占用4位坤候,所以每個byte可以用兩個字符來表示白筹,反之亦然。舉個栗子
byte = 123
用二進制表示:0111 1011
每4位用字符表示: 7 b
注意:java是用補碼來進行二進制計算的徒河,因為上面最高位為0送漠,即為正數(shù),而正數(shù)的補碼為自身闽寡,所以沒什么問題纵穿,下面看看負數(shù)的栗子:
16位進制字符串表示: a b
用二進制表示:1010 1011
二進制補碼: 1101 0101
byte:-85 (如果不用補碼計算奢人,應(yīng)該為171淆院,超出byte的范圍了)
是的,原理就這么簡單支救,接下來用代碼實現(xiàn):
- byte[] 轉(zhuǎn)16進制字符串
法1
思路:先把byte[] 轉(zhuǎn)換維char[]拷淘,再把char[] 轉(zhuǎn)換為字符串
public static String bytes2Hex(byte[] src) {
if (src == null || src.length <= 0) {
return null;
}
char[] res = new char[src.length * 2]; // 每個byte對應(yīng)兩個字符
final char hexDigits[] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };
for (int i = 0, j = 0; i < src.length; i++) {
res[j++] = hexDigits[src[i] >> 4 & 0x0f]; // 先存byte的高4位
res[j++] = hexDigits[src[i] & 0x0f]; // 再存byte的低4位
}
return new String(res);
}
法2
思路:先把byte轉(zhuǎn)換為int類型,再轉(zhuǎn)換為字符串
public static String bytes2Hex(byte[] src){
if (src == null || src.length <= 0) {
return null;
}
StringBuilder stringBuilder = new StringBuilder("");
for (int i = 0; i < src.length; i++) {
// 之所以用byte和0xff相與贬堵,是因為int是32位结洼,與0xff相與后就舍棄前面的24位,只保留后8位
String str = Integer.toHexString(src[i] & 0xff);
if (str.length() < 2) { // 不足兩位要補0
stringBuilder.append(0);
}
stringBuilder.append(str);
}
return stringBuilder.toString();
}
- 16進制字符串轉(zhuǎn)byte[]
思路:先把字符串轉(zhuǎn)換為char[]蒸殿,再轉(zhuǎn)換為byte[]鸣峭。
因為兩個字符對應(yīng)一個byte,所以字符串的長度不能為奇數(shù)喔(哪位有想到好辦法解決這一問題的摊溶,求告知)。
public static byte[] hex2Bytes(String hexString) {
if (hexString == null || hexString.equals("")) {
return null;
}
int length = hexString.length() / 2;
char[] hexChars = hexString.toCharArray();
byte[] bytes = new byte[length];
String hexDigits = "0123456789abcdef";
for (int i = 0; i < length; i++) {
int pos = i * 2; // 兩個字符對應(yīng)一個byte
int h = hexDigits.indexOf(hexChars[pos]) << 4; // 注1
int l = hexDigits.indexOf(hexChars[pos + 1]); // 注2
if(h == -1 || l == -1) { // 非16進制字符
return null;
}
bytes[i] = (byte) (h | l);
}
return bytes;
}
注:注1得到xxxx0000盖腕,注2得到0000xxxx浓镜,相或就把兩個字符轉(zhuǎn)換為一個byte了。
- 再舉個栗子
md5加密
public static String getMd5ByFile(File file) {
String ret= null;
FileInputStream fis = null;
try {
fis = new FileInputStream(file);
MessageDigest md = MessageDigest.getInstance("MD5");
byte[] buffer = new byte[1024];
int len;
while((len = fis.read(buffer)) > 0) {
md.update(buffer, 0, len);
}
ret = bytes2Hex(md.digest()); // 把md5加密后的byte[]轉(zhuǎn)換為字符串
} catch (Exception e) {
e.printStackTrace();
} finally {
if(fis != null) {
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
return ret;
}
寫在最后
好了听隐,應(yīng)該懂了吧哄啄,其實并不難的风范。上面的是我個人的理解沪么,難免有錯。若有錯禽车,歡迎指正。
如果這篇文章對你有幫助的話州胳,不妨點個喜歡唄~