Shiro 框架的MD5加密算法实现原理

直接上代码:该代码可以直接用于项目中做MD5加密,加盐加密,多层散列加密

import java.io.UnsupportedEncodingException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;

public class ShiroMd5 {
    private static final char[] DIGITS = new char[]{'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};

    public static void main(String[] args) {
        ShiroMd5 shiroMd5 = new ShiroMd5();
        String pwd = "123456";
        String salt = "saltTest";
        int hashNumber =1;
        String md5SaltHash = shiroMd5.getMd5SaltHash(pwd, salt, hashNumber);
        System.out.println(md5SaltHash);
    }

    /**
     * 加密步骤:
     * @param pwd
     * @param salt
     * @param hashNumber
     * @return
     * 说明:shiro 源码提供的方法支持多种类型变量加密:这里只使用String类型 String pwd, String salt
     */
    public String getMd5SaltHash(String pwd, String salt, int hashNumber){
        if(pwd == null || salt == null) return null;
        try {
            //shiro 官方默认是使用UTF-8编码格式
            byte[] pwdBytes = pwd.getBytes("UTF-8");
            byte[] saltBytes = salt.getBytes("UTF-8");
            //加密规则:先加盐后加密,再多次散列加密
            byte[] hash = hash(pwdBytes, saltBytes, hashNumber);
            //生成的hash 是16位的byte;通过encode 方法对每一位(byte)对应生成两位char
            char[] encode = encode(hash);
            //char 转 String
            return new String(encode);

        } catch (UnsupportedEncodingException e) {
            e.printStackTrace();
        }

        return null;
    }

    /**
     * 使用JDK 自带的MessageDigest 加密实现
     * @param bytes
     * @param salt
     * @param hashIterations
     * @return
     */
    protected byte[] hash(byte[] bytes, byte[] salt, int hashIterations) {
        //java.security.MessageDigest: jdk 自带的类
        MessageDigest digest = this.getDigest("MD5");
        //加盐:digest.update(salt);
        if (salt != null) {
            digest.reset();
            digest.update(salt);
        }
        //第一次散列加密: digest.digest(bytes);
        byte[] hashed = digest.digest(bytes);
        //多次散列加密:判断
        int iterations = hashIterations - 1;
        for(int i = 0; i < iterations; ++i) {
            digest.reset();
            hashed = digest.digest(hashed);
        }
        return hashed;
    }

    /**
     * 创建MessageDigest 加密类
     * @param algorithmName
     * @return
     */
    protected MessageDigest getDigest(String algorithmName) {
        try {
            return MessageDigest.getInstance(algorithmName);
        } catch (NoSuchAlgorithmException var4) {
            String msg = "No native '" + algorithmName + "' MessageDigest instance available on the current JVM.";
            return null;
        }
    }

    /**
     *
     * @param data
     * @return
     */
    public static char[] encode(byte[] data) {
        int l = data.length;
        char[] out = new char[l << 1];//相当于l 乘于 2,位运算速度更快,更符合机器运算规则
        int i = 0;
        //byte 中的所有值,共16 位,值的范围:-128~127 (2^8)一个字节8位
        System.out.println(Arrays.toString(data));
        //data 为16 位,每个byte 对应生成两个char;最终的md5 是32位。
        //计算方式
        for(int n = 0; i < l; ++i) {
            out[n++] = DIGITS[(0xf0 & data[i]) >>> 4];//"无符号"右移运算 。先“与”运算,再右移三位,最后取DIGITS常量数组中的字符
            out[n++] = DIGITS[0xf & data[i]];//只“与”运算,最后取DIGITS常量数组中的字符
            /*--------------格式化输出看看---------S-----*/
            System.out.print((0xf0 & data[i]) >>> 4);
            System.out.print(" ");
            System.out.print(0xf & data[i]);
            System.out.print(" ");
            //可以看到32位都是正整数,值的范围:0~15,十进制表示16进制的数,通过常量DIGITS[]转成16进制数
            /*--------------格式化输出看看----------E----*/
        }
        System.out.println("\n"+Arrays.toString(out));//最终可以看到32位16进制的字符串:MD5值
        return out;
    }
}

结果输出:

 

posted @ 2020-08-11 23:12  渊渟岳  阅读(402)  评论(0编辑  收藏  举报