Java生成MD5值

MD5是什么

message-digest algorithm 5(信息-摘要算法)。经常说的“MD5加密”,就是它→信息-摘要算法。

在下载一下东西时,经常在一些压缩包属性里,看到md5值。而且这个下载页面,很可能会在某一个地方,写了一句,此文件的MD5值为XXXXXXXXX。这有什么作用呢?

白话白话:md5,其实就是一中算法。可以将一个字符串,或文件,或压缩包,执行md5后,就可以生成一个固定长度为128bit的串。这个串,基本上是唯一的。

所以,有人修过压缩包后,就会生成新的串,这时就可以拿网站提供的串和新生成的串对比,如果不同,那就是被人修过过了。

 

加密和摘要,是不一样的

加密后的消息是完整的;具有解密算法,得到原始数据;

摘要得到的消息是不完整的;通过摘要的数据,不能得到原始数据;

所以,注意 “加密“解密”的说法不够正确。

QQ截图20150311162809

 

MD5长度

有人说md5,128位,32位,16位,到底md5多长?

md5的长度,默认为128bit,也就是128个0和1的二进制串。

这样表达是很不友好的。

所以将二进制转成了16进制,每4个bit表示一个16进制,

所以128/4 = 32 换成16进制表示后,为32位了。

QQ截图20150305111855

 

为什么网上还有md5是16位的呢?

网上有很多帖子,md5 32位 16位 加密 区别。

仔细观察admin生成的32位和16位的md5值……

查询结果:

md5(admin,32) = 21232f297a57a5a743894a0e4a801fc3

md5(admin,16) = 7a57a5a743894a0e

看出来了吧!

其实16位的长度,是从32位md5值来的。是将32位md5去掉前八位,去掉后八位得到的。

 

MD5的作用

①一致性检验,最上面那个例子

②数字签名,还是最上面那个例子。只是把md5看出了一个指纹,按了个手印说明独一无二了。

③安全访问认证,这个就是平时系统设计的问题了。

在用户注册时,会将密码进行md5加密,存到数据库中。这样可以防止那些可以看到数据库数据的人,恶意操作了。

 

md5不能破解吗?

md5是不可逆的,也就是没有对应的算法,从生产的md5值逆向得到原始数据。

但是如果使用暴力破解,那就另说了。

 

md5是唯一的吗?

md5作为数据库中的主键可行吗?这就涉及到一个问题,md5值是唯一的吗?答案是,不唯一。

也就是一个原始数据,只对应一个md5值;

但是一个md5值,可能对应多个原始数据。

 

java中生成MD5的值

import java.security.MessageDigest;

/**
 * message-digest algorithm 5(信息-摘要算法)
 * 
 * md5的长度,默认为128bit,也就是128个 0和1的 二进制串 。
 * 
 * 128/4 = 32 换成 16进制 表示后,为32位了。
 */
public class MD5Util {

    // 测试方法
    public static void main(String[] args) {
	String pwd = "123456";
	System.out.println("加密前: " + pwd);
	System.err.println("加密后: " + MD5Util.getMD5(pwd));
    }

    /**
     * 生成md5
     * 
     * @param message
     * @return
     */
    public static String getMD5(String message) {
	String md5str = "";
	try {
	    // 1 创建一个提供信息摘要算法的对象,初始化为md5算法对象
	    MessageDigest md = MessageDigest.getInstance("MD5");

	    // 2 将消息变成byte数组
	    byte[] input = message.getBytes();

	    // 3 计算后获得字节数组,这就是那128位了
	    byte[] buff = md.digest(input);

	    // 4 把数组每一字节(一个字节占八位)换成16进制连成md5字符串
	    md5str = bytesToHex(buff);

	} catch (Exception e) {
	    e.printStackTrace();
	}
	return md5str;
    }

    /**
     * 二进制转十六进制
     * 
     * @param bytes
     * @return
     */
    public static String bytesToHex(byte[] bytes) {
	StringBuffer md5str = new StringBuffer();
	// 把数组每一字节换成16进制连成md5字符串
	int digital;
	for (int i = 0; i < bytes.length; i++) {
	    digital = bytes[i];

	    if (digital < 0) {
		digital += 256;
	    }
	    if (digital < 16) {
		md5str.append("0");
	    }
	    md5str.append(Integer.toHexString(digital));
	}
	return md5str.toString().toUpperCase();
    }
}

 

 转载自大牛笔记 --- http://www.weixuehao.com


<think>好的,我现在要回答用户的问题:Java中如何对固定字符串生成一致的MD5,并涉及哈希一致性的算法行为。首先,我需要回忆MD5的基本原理和Java中的实现方式。 用户提到“固定字符串生成MD5的一致性”,可能关心的是每次对同一个字符串使用MD5算法是否能得到相同的哈希。根据MD5的特性,相同的输入应该总是产生相同的输出,所以理论上是一致的。但需要确保实现正确,比如字符编码是否正确处理,比如是否将字符串转换为字节数组时使用了正确的字符集(如UTF-8)。 接下来,我需要参考用户提供的引用。引用[1]提到MD5算法原理和C#示例,但用户需要Java的实现。引用[2]是Java的字符串哈希算法,但这是另一个算法(可能是Java默认的hashCode方法),不是MD5。引用[3]讨论暴雪哈希算法,可能与用户的问题无关。因此,可能需要结合通用知识来回答。 在Java中,生成MD5通常使用MessageDigest类。步骤大致是:获取MD5实例,用update方法传入字节数组,然后调用digest方法得到哈希,再将字节数组转换为十六进制字符串。需要注意字符编码,例如使用getBytes("UTF-8")而不是依赖平台默认编码,否则不同环境可能导致不同结果。 用户可能还关心哈希一致性的问题,比如不同的字符串是否可能产生相同的MD5(碰撞)。MD5已经被证明存在碰撞,不适用于安全场景,但如果是固定字符串,只要实现正确,同一字符串的MD5应该一致。 现在需要验证是否正确。例如,Java中是否必须指定字符编码?如果用户不指定,使用默认的可能会出问题。比如,字符串在不同操作系统上默认编码不同,导致字节数组不同,从而MD5不同。因此,正确的做法是明确指定编码,如UTF-8。 另外,生成MD5字节数组需要正确转换为十六进制字符串。常见的错误是直接调用toString或者使用不正确的转换方法,导致结果不一致。例如,应该将每个字节转换为两位十六进制,必要时补零。 总结回答步骤: 1. 使用MessageDigest获取MD5实例。 2. 将字符串转换为字节数组,指定统一编码(如UTF-8)。 3. 调用digest方法得到哈希字节数组。 4. 将字节数组转换为十六进制字符串。 可能的问题包括:未处理异常、编码不一致、转换错误。示例代码需要处理这些点。例如,使用try-catch处理可能的NoSuchAlgorithmException(虽然MD5应该总是存在),以及UnsupportedEncodingException(但UTF-8是标准编码)。 最后,用户可能想知道为何需要哈希一致性,比如在分布式系统中用于数据分片或缓存键。这时候需要提到MD5的特性,如确定性输出,但也要指出其安全性问题,建议在安全场景使用更安全的算法如SHA-256。</think>在Java中,对固定字符串生成一致的MD5需要遵循以下步骤,确保算法行为和编码方式统一: ### 1. MD5算法实现步骤 - **字符编码统一**:将字符串转换为字节数组时,需明确指定字符编码(如`UTF-8`),避免因系统默认编码差异导致结果不一致。 - **使用`MessageDigest`类**:Java通过`java.security.MessageDigest`提供MD5算法支持。 - **字节数组转十六进制**:将生成MD5字节数组转换为固定长度的十六进制字符串。 ### 2. 示例代码 ```java import java.security.MessageDigest; import java.security.NoSuchAlgorithmException; public class MD5Example { public static String generateMD5(String input) { try { MessageDigest md = MessageDigest.getInstance("MD5"); byte[] inputBytes = input.getBytes("UTF-8"); // 指定编码 byte[] hashBytes = md.digest(inputBytes); // 转换为十六进制字符串 StringBuilder hexString = new StringBuilder(); for (byte b : hashBytes) { String hex = String.format("%02x", b & 0xff); hexString.append(hex); } return hexString.toString(); } catch (NoSuchAlgorithmException | java.io.UnsupportedEncodingException e) { throw new RuntimeException("MD5算法不支持或编码错误", e); } } public static void main(String[] args) { String str = "固定字符串"; System.out.println("MD5: " + generateMD5(str)); // 输出示例: MD5: 3e25960a79dbc69b674cd4ec67a72c62 } } ``` ### 3. 哈希一致性说明 - **确定性输出**:同一字符串在相同编码下生成MD5始终一致[^1]。 - **碰撞风险**:MD5存在已知的哈希碰撞问题,不适用于安全性要求高的场景[^3]。 - **编码影响**:若未明确指定编码(如`UTF-8`),不同操作系统的默认编码可能导致结果差异。 ### 4. 优化与注意事项 - **替换更安全算法**:如需防碰撞或安全性,建议使用`SHA-256`(将`MessageDigest.getInstance("MD5")`改为`"SHA-256"`)。 - **性能**:MD5计算效率较高,适合非敏感场景的快速哈希生成[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值