安全算法(一)数字摘要

  数字摘要,也叫消息摘要。它是一种单向的,不可逆的加密算法。发送方:采用单向hash函数对消息进行计算,得到摘要。发送消息和摘要。接收方:把接收的消息,按着同样的hash函数计算,新产生的摘要和发送来的原摘要进行对比。如果2个摘要不一致,说明消息不完整了。

 

  其中存在hash碰撞,即:2条消息不相同,但hash函数计算出的摘要相同。所以说hash函数的好坏,是由发生碰撞的概率决定的。如果攻击者可以构造出具有相同hash值的消息,那这个hash函数很危险咯。


  消息摘要的特点:

1、无论输入的消息有多长,计算出来的摘要长度是固定的。

2、消息摘要具备的基本性质:不同的消息,计算出来的摘要不同;相同的消息,计算出来的摘要相同;

3、消息摘要是单向、不可逆的。从计算出的摘要中,不能逆向得到消息内容。

 

  常用的消息摘要算法实现:

一、MD5

  Message Digest Algorithm5,这是数字摘要算法很常见的一种实现。用于确保信息传输的完整性和一致性,摘要长度为128位。MD5是由MD4MD3MD2改进而来,主要增强了算法的复杂度和不可逆性。

  基于JavaMD5算法实现:

  

public static byte[] testMD5(String content) {
        MessageDigest md = null;
        byte[] result=null;
        try {
            // 生成一个MD5加密计算摘要
            md = MessageDigest.getInstance("MD5");
            logger.info("MD5摘要长度:" + md.getDigestLength());
            result = md.digest(content.getBytes("utf8"));

        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        logger.info("MD5摘要内容"+byte2Hex(result));
        return  result;
    }

 

二、SHA

  全称是Secure Hash Algorithm,即安全散列算法,通常称为SHA-1,是基于MD4算法。它生成的摘要长度是160位,由于生成摘要更长,运算过程更加复杂,在相同硬件下,SHA-1的运行速度慢于MD5,但是也更安全。

  基于JavaSHA-1算法实现:

public static byte[] testSHA(String content){
        byte[] result=null;
        try {
            MessageDigest md = MessageDigest.getInstance("SHA-1");
            result= md.digest(content.getBytes("utf8"));
            logger.info("SHA摘要内容"+byte2Hex(result));

        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }
        return  result;
    }


  问题:hash函数计算出的摘要是byte[]字节,需要我们转换成16进制的字符串。

/**
     * byte[]字节转16进制字符串
     * @param bytes 要转换的byte[]
     * @return
     */
    public static String byte2Hex(byte[] bytes){
        StringBuilder hex = new StringBuilder();
        for (int i = 0; i < bytes.length; i++) {
            byte b = bytes[i];
            boolean negative = false;   //是否为负数
            if (b < 0) {
                negative = true;
            }
            int inte= Math.abs(b);
            if (negative)inte = inte | 0x80;    //负数会转成正数(最高位的负号变成数值计算)
            String temp = Integer.toHexString(inte & 0xFF);
            if(temp.length() == 1){
                hex.append("0");
            }
            hex.append(temp.toLowerCase());
        }
        return hex.toString();
    }



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值