java - 關(guān)于字符串編碼。
問題描述
public static void main(String[] args) throws Exception { String str = 'resource'; System.out.println(Arrays.toString(getHash(str,'MD5').getBytes()));//使用默認解碼后輸出}public static String getHash(String str, String hashType) { try {MessageDigest digest = MessageDigest.getInstance(hashType);digest.reset();byte[] b = digest.digest(str.getBytes());System.out.println(Arrays.toString(b)); //編碼前輸出return new String(b); //使用默認編碼 } catch (NoSuchAlgorithmException e) {e.printStackTrace(); } return str;}
輸出:[-106, -85, 78, 22, 63, 78, -32, 58, -86, 77, 16, 81, -86, 81, -46, 4]
[-17, -65, -67, -17, -65, -67, 78, 22, 63, 78, -17, -65, -67, 58, -17, -65, -67, 77, 16, 81, -17, -65, -67, 81, -17, -65, -67, 4]
為什么編碼前和編碼后再解碼所輸出的不一樣?
問題解答
回答1:你可能認為(new String(b)).getBytes().equals(b),實際上并非如此。(盡管new String(s.getBytes()).equals(s)一定是。)
因為byte[]轉(zhuǎn)換成String時,有些字節(jié)是未必能轉(zhuǎn)換成字符的,比如第一個-106、第二個-85就是,所以轉(zhuǎn)換成String時前兩個就變成了未知字符(表面上會顯示?,但實際上是一個Unicode字符),再轉(zhuǎn)成byte[](你這邊defaultCharset應(yīng)該是UTF-8吧),每個未知字符就變成3個字節(jié)了。
如果用GBK的話,情況還算好,但還是略有不同:
[-106, -85, 78, 22, 63, 78, -32, 58, -86, 77, 16, 81, -86, 81, -46, 4][-106, -85, 78, 22, 63, 78, 63, 58, -86, 77, 16, 81, -86, 81, 63, 4]
所以結(jié)論是:如果用String表示一個Hash值,不能把byte[]強轉(zhuǎn)換成String,而是按慣例轉(zhuǎn)換成16進制表示。
相關(guān)文章:
1. Java反射問題:為什么android.os.Message的recycleUnchecked方法不能通過反射獲取到?2. 如何分別在Windows下用Winform項模板+C#,在MacOSX下用Cocos Application項目模板+Objective-C實現(xiàn)一個制作游戲的空的黑窗口?3. html5和Flash對抗是什么情況?4. php如何獲取訪問者路由器的mac地址5. javascript - 在 vue里面用import引入js文件,結(jié)果為undefined6. javascript - ...mapGetters和...mapState獲取到的state,怎么拿來在methods中操作?7. 小程序怎么加外鏈,語句怎么寫!求救新手,開文檔沒發(fā)現(xiàn)8. python - linux怎么在每天的凌晨2點執(zhí)行一次這個log.py文件9. javascript - vue-resource中如何設(shè)置全局的timeout?10. 前端 - node vue webpack項目文件結(jié)構(gòu)
