简单实现如下:
public class MD5Util { public static String md5(String txt) { try { MessageDigest md = MessageDigest.getInstance("MD5"); // 问题主要出在这里,Java的字符串是unicode编码,不受源码文件的编码影响;而PHP的编码是和源码文件的编码一致,受源码编码影响。 md.update(txt.getBytes("GBK")); StringBuffer buf = new StringBuffer(); for (byte b : md.digest()) { buf.append(String.format("%02x", b & 0xff)); } return buf.toString(); } catch (Exception e) { e.printStackTrace(); return null; } } }
介绍下MD5实现的一种简要流程,MD5算法是将字符串转化为32或64为的hash值,而这个值只要稍有不同结果就会大不相同,而hash又是著名的散列算法,显然密文和明文不是一一对应的情况,使用时常和对称加密和非对称配合,主要起的是一个验签的作用。因为MD5的算法是不可逆向的,比如将服务器A将一串字符串和MD5加盐生成的哈希值传给服务器B,添加的盐值是双方事先约定好的,B接受到了数据通过将传送的字符串加盐与收到的哈希值比对,就可以知道有没有受到中途篡改。
这个算法关键在于两个地方,“%02x”指的是将字节以十六进制二位的形式保存,而在运算时会计算最高位的0,0的符号位表示正数,如果传入的是负数就会导致符号错乱,与 0xff与操作后,0xff都为1,符号会保存下来。
计算机有补码的存储方式,如果以int型为例,int为4位字节,每字节占8位,总共为2^32 如果是有符号位 则正负占一位,则正数为 231负数为231+1 ,因为显然 符号位为负的0是没有意义的,所以符号位为1的0会表示-2^31+1这个值。