java實現MD5隨機鹽加密


記錄一下實現的MD5隨機鹽加密

/**
 *  動態鹽的MD5加密
 * @author tireless
 */
public class Md5Util {
    

    /**
     * 普通MD5,只是實現下,不推薦使用,是不可逆的,但是聰明的人想到了查表,導致普通MD5的安全壁壘 GG 了;
     * @author tireless
     * @time 2022年
     * @return 加密的字符
     */
    public static String MD5(String input) {
        MessageDigest md5 = null;
        try {
            md5 = MessageDigest.getInstance("MD5");
        } catch (NoSuchAlgorithmException e) {
            return "JDK不支持該算法,檢查下JDK";
        } catch (Exception e) {
            e.printStackTrace();
            return "";
        }

        byte[] byteArray = input.getBytes();

        byte[] md5Bytes = md5.digest(byteArray);

        StringBuilder hexValue = new StringBuilder();
        //     將加密完的字符串,全部轉成0-9、a-f的字符串
        for (int i = 0; i < md5Bytes.length; i++) {
            int val = ((int) md5Bytes[i]) & 0xff;
            //     如果小於16,也就是16進制只有1位的情況下。前面補0
            if (val < 16) {
                hexValue.append("0");
            }
            hexValue.append(Integer.toHexString(val));
        }
        return hexValue.toString();

    }


    /**
     * 加鹽MD5
     *
     * @param password 原始密碼
     * @return 加鹽的MD5字符串
     * @author tireless
     *
     */
    public static String generate(String password) {
//        生成隨機鹽,長度12位
        byte[] bytes = new byte[12];
        SecureRandom random = new SecureRandom();
        random.nextBytes(bytes);

        StringBuilder builder = new StringBuilder();
//        將字節數組變為字符串
        for (int i = 0; i < bytes.length; i++) {
//            將生成的值,全部映射到0-255 之間
            int val = ((int) bytes[i]) & 0xff;
            if (val < 16) {
//                為了控制鹽的長度,這里小於16 的值,我們將它補為 大於16的值;
//                這樣,生的鹽的長度是固定的:bytes * 2 ;
                builder.append(Integer.toHexString(val + 16));
            } else {
                builder.append(Integer.toHexString(val));
            }
        }

//        最終的鹽,長度是 12*2 = 24 ;
        String salt = builder.toString();


//        先加鹽Md5一把,再將 MD5 轉換成 24位的 base64 位編碼
        password = md5Hex(password + salt);

        char[] cs = new char[salt.length() + password.length()];

        for (int i = 0; i < cs.length; i += 4) {

//            密碼編碼
            cs[i] = password.charAt(i / 2);
            cs[i + 2] = password.charAt(i / 2 + 1);
//            鹽編碼
            cs[i + 1] = salt.charAt(i / 2);
            cs[i + 3] = salt.charAt(i / 2 + 1);

        }
        return new String(cs);
    }

    /**
     * 校驗加鹽后是否和原文一致
     *
     * @param password
     * @param md5
     * @return true 代表密碼驗證通過
     * @author tireless
     */
    public static boolean verify(String password, String md5) {
//        解碼密碼
        char[] cs1 = new char[24];
//        解碼鹽
        char[] cs2 = new char[24];
//        從MD5 中取出鹽
        for (int i = 0; i < md5.length(); i += 4) {
//            取出鹽
            cs2[i / 2] = md5.charAt(i + 1);
            cs2[i / 2 + 1] = md5.charAt(i + 3);
//            取出密碼的MD5值(經過Base64轉換后的MD5)
            cs1[i / 2] = md5.charAt(i + 0);
            cs1[i / 2 + 1] = md5.charAt(i + 2);
        }

        String salt = new String(cs2);
        System.out.println(md5Hex(password + salt));

        return md5Hex(password + salt).equals(new String(cs1));
    }

    /**
     * 獲取十六進制字符串形式的MD5摘要
     */
    private static String md5Hex(String src) {
        try {
            MessageDigest md5 = MessageDigest.getInstance("MD5");
            byte[] bs = md5.digest(src.getBytes());
            return new String(Base64.getEncoder().encode(bs));
        } catch (Exception e) {
            return null;
        }
    }

}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM