nighty

折腾的年华
posts - 37, comments - 143, trackbacks - 0, articles - 0

Java MD5加密生成摘要

Posted on 2008-05-08 15:02 寒武纪 阅读(6530) 评论(2)  编辑  收藏 所属分类: 信息安全

    刚好最近项目中需要用到一点加密的东西,java安全类库提供了一个java.security.MessageDigest类,此 MessageDigest 类为应用程序提供信息摘要算法的功能,如 MD5 或 SHA 算法。信息摘要是安全的单向哈希函数,它接收任意大小的数据,并输出固定长度的哈希值。有现成的当然是最好的,省事省力。
    MD5的非常有实际应用性。有网友给出这样的描述,可以参照一下:http://blog.csdn.net/Daping_Zhang/archive/2005/05/28/382688.aspx
     该类的getInstance(String algorithm) 方法返回一个MessageDigest的实体,加密的一系统的digest()方法和update(byte input)方法。加密后返回一个byte[],16位,我们经常见到很多开源网站的下载地址会有一个[md5]的链接,打开其实就是一小段文本内容。例如:
    MD5 (commons-logging-1.1.1-bin.zip) = f88520ed791673aed6cc4591bc058b55
    这是Jakarta的logging组件下载时提供的MD5摘要信息,是对这个zip包进行全文加密生成的摘要,摘要码就是后面的f88520ed791673aed6cc4591bc058b55,如果你下载以后,按照MD5的算法生成自己的摘要,如果这二个摘要一样,就证明这个文件是没有被人篡改过的。
    遇到的问题是Java的MessageDigest类执行后返回的byte[16]得转换成十六进制的字符串,如果直接用new String(byte[]),得到的结果将是不正确的。算法有很多网友提供了,照搬了。比较有趣的是,commons-logging提供的那个MD5居然和我自己生成的不一样(难道文件被修改过?),后来尝试了其它地方提供的MD5码,都没有问题。
    有很多相关的现成代码,搜集了一下整理如下(经过验证):

public class MD5Builder {

    
static Logger logger = Logger.getLogger(MD5Builder.class);
     
// 用来将字节转换成 16 进制表示的字符
    static char hexDigits[] = '0''1''2''3''4''5''6''7''8',
            
'9''a''b''c''d''e''f' }

    
    
/**
     * 对文件全文生成MD5摘要
     * 
@param file   要加密的文件
     * 
@return MD5摘要码
     
*/

    
public static String getMD5(File file) {
        FileInputStream fis 
= null;
        
try {
            MessageDigest md 
= MessageDigest.getInstance("MD5");

            logger.info(
"MD5摘要长度:" + md.getDigestLength());
            fis 
= new FileInputStream(file);
            
byte[] buffer = new byte[2048];
            
int length = -1;
            logger.info(
"开始生成摘要");
            
long s = System.currentTimeMillis();
            
while ((length = fis.read(buffer)) != -1{
                md.update(buffer, 
0, length);
            }

            logger.info(
"摘要生成成功,总用时: "
                    
+ (System.currentTimeMillis() - s) + "ms");
            
byte[] b = md.digest();
            
return byteToHexString(b);
            
// 16位加密
            
// return buf.toString().substring(8, 24);
        }
 catch (Exception ex) {
            logger.error(ex);
            ex.printStackTrace();
            
return null;
        }
finally {
            
try {
                fis.close();
            }
 catch (IOException ex) {
                ex.printStackTrace();
            }

        }

    }


    
/**
     * 对一段String生成MD5加密信息
     * 
@param message 要加密的String
     * 
@return 生成的MD5信息
     
*/

    
public static String getMD5(String message){
        
try {
            MessageDigest md 
= MessageDigest.getInstance("MD5");
            logger.info(
"MD5摘要长度:" + md.getDigestLength());
            
byte[] b = md.digest(message.getBytes());
            
return byteToHexString(b);
        }
 catch (NoSuchAlgorithmException e) {
            logger.error(e);
            e.printStackTrace();
            
return null;
        }

    }

    
    
/**
     * 把byte[]数组转换成十六进制字符串表示形式
     * 
@param tmp    要转换的byte[]
     * 
@return 十六进制字符串表示形式
     
*/

    
private static String byteToHexString(byte[] tmp) {
        String s;
        
// 用字节表示就是 16 个字节
        char str[] = new char[16 * 2]; // 每个字节用 16 进制表示的话,使用两个字符,
        
// 所以表示成 16 进制需要 32 个字符
        int k = 0// 表示转换结果中对应的字符位置
        for (int i = 0; i < 16; i++// 从第一个字节开始,对 MD5 的每一个字节
            
// 转换成 16 进制字符的转换
            byte byte0 = tmp[i]; // 取第 i 个字节
            str[k++= hexDigits[byte0 >>> 4 & 0xf]; // 取字节中高 4 位的数字转换, 
            
// >>> 为逻辑右移,将符号位一起右移
            str[k++= hexDigits[byte0 & 0xf]; // 取字节中低 4 位的数字转换
        }

        s 
= new String(str); // 换后的结果转换为字符串
        return s;
    }

}

  



刚进场的时候戏就落幕

Feedback

# re: Java MD5加密生成摘要[未登录]  回复  更多评论   

2009-03-13 20:46 by aaa
dddddddddddddd

# re: Java MD5加密生成摘要  回复  更多评论   

2009-03-20 15:33 by 书页
/** *//**
* 对一段String生成MD5加密信息
* @param message 要加密的String
* @return 生成的MD5信息
*/
public static String getMD5(String message){
try {
MessageDigest md = MessageDigest.getInstance("MD5");
logger.info("MD5摘要长度:" + md.getDigestLength());
byte[] b = md.digest(message.getBytes());
return byteToHexString(b);
} catch (NoSuchAlgorithmException e) {
logger.error(e);
e.printStackTrace();
return null;
}
}



byte[] b = md.digest(message.getBytes());
应该修改成:
byte[] b = md.digest(message.getBytes("GBK"));

指定编码,这样这块代码在不同编码的系统下出来的MD5就是一样啦!

只有注册用户登录后才能发表评论。


网站导航: