3Des加密不同JDK版本,加密结果有差异

发布于 2022-01-03 12:27:02 字数 262 浏览 873 评论 3

银行项目

用了几个加密的工具类,里面很杂,各种加密方法。

所以想调方法本身的问题,难度很大。

现在存在一个问题,就在原本使用JDK6的时候,加/解密钥的时候,结果都是正常的。

但是现在因为项目原因,一定要使用JDK8,发现在JDK8下,加解密结果跟JDK6的有差异,密钥全部比对不上。

这是为什么?或者怎么解决。

主要是8583报文中的数据加解密,使用Desc、3Des、AES几种算法

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

想挽留 2022-01-07 18:48:24

怎么解决的的呢必须换jdk吗,我们和第三方版本不一致

浮生未歇 2022-01-07 13:45:12

引用来自“BoXuan”的评论

给你一个我用的,看是否有帮助。再就是你看看你是否用了随机秘钥源,可能问题出在这里。

JAVA默认用ECB模式,但当用CBC模式下,如果还是用SecureRandom, 则每次加密的结果都会不一样,因为JAVA内部会用随机的IV来初始化Cipher对象,如示例代码, 由于Cipher.getInstance("DES/CBC/PKCS5Padding")使用了CBC, 因此我这里用的javax.crypto.spec.IvParameterSpec包下的IvParameterSpec来初始化向量IV
JAVA中默认的算法模式为ECB,默认填充方式为PKCS5Padding。我这里因为是跟unity通信,而C#默认是CBC模式,所以此处都是用的CBC模式

public class DES {

    private DES() {
    }

    /**
     * 加密
     *
     * @param src byte[] 加密的数据源
     * @param password byte[] 加密秘钥
     * @return byte[] 加密后的数据
     */
    public static byte[] encrypt(byte[] src, byte[] password) {
        try {
            DESKeySpec desKey = new DESKeySpec(password);
            //创建一个密匙工厂,然后用它把DESKeySpec转换成
            SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
            SecretKey securekey = keyFactory.generateSecret(desKey);
            //Cipher对象实际完成加密操作
            Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
            //ECB模式下,iv不需要
            IvParameterSpec iv = new IvParameterSpec(password);
            //用密匙初始化Cipher对象
            cipher.init(Cipher.ENCRYPT_MODE, securekey, iv);
            //现在,获取数据并加密
            //正式执行加密操作
            return cipher.doFinal(src);
        } catch (InvalidKeyException | NoSuchAlgorithmException | InvalidKeySpecException | NoSuchPaddingException | IllegalBlockSizeException | BadPaddingException | InvalidAlgorithmParameterException ex) {
            Logger.getLogger(DES.class.getName()).log(Level.SEVERE, null, ex);
        }
        return null;
    }

    /**
     * 加密
     *
     * @param src byte[] 加密的数据源
     * @param password String 加密秘钥
     * @return byte[] 加密后的数据
     */
    public static byte[] encrypt(byte[] src, String password) {
        byte[] keyBytes = password.getBytes(Charset.utf8);
        return encrypt(src, keyBytes);
    }

    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password byte[] 解密秘钥
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, byte[] password) {
        try {
            // DES算法要求有一个可信任的随机数源
            //SecureRandom random = new SecureRandom();
            // 创建一个DESKeySpec对象
            DESKeySpec desKey = new DESKeySpec(password);
            // 创建一个密匙工厂
            SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
            // 将DESKeySpec对象转换成SecretKey对象
            SecretKey securekey = keyFactory.generateSecret(desKey);
            // Cipher对象实际完成解密操作,CBC为加密模式,
            Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
            //ECB模式下,iv不需要
            IvParameterSpec iv = new IvParameterSpec(password);
            // 用密匙初始化Cipher对象
            cipher.init(Cipher.DECRYPT_MODE, securekey, iv);
            // 真正开始解密操作
            return cipher.doFinal(src);
        } catch (InvalidKeyException | NoSuchAlgorithmException | InvalidKeySpecException | NoSuchPaddingException | InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException ex) {
            Logger.getLogger(DES.class.getName()).log(Level.SEVERE, null, ex);
        }
        return null;
    }
    
    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password String 解密秘钥
     * @param psdCharset 秘钥字符编码
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, String password, java.nio.charset.Charset psdCharset) {
        byte[] keyBytes = password.getBytes(psdCharset);
        return decrypt(src, keyBytes);
    }

    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password String 解密秘钥
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, String password) {
        return decrypt(src, password, Charset.utf8);
    }
}

离去的眼神 2022-01-05 14:17:11

给你一个我用的,看是否有帮助。再就是你看看你是否用了随机秘钥源,可能问题出在这里。

JAVA默认用ECB模式,但当用CBC模式下,如果还是用SecureRandom, 则每次加密的结果都会不一样,因为JAVA内部会用随机的IV来初始化Cipher对象,如示例代码, 由于Cipher.getInstance("DES/CBC/PKCS5Padding")使用了CBC, 因此我这里用的javax.crypto.spec.IvParameterSpec包下的IvParameterSpec来初始化向量IV
JAVA中默认的算法模式为ECB,默认填充方式为PKCS5Padding。我这里因为是跟unity通信,而C#默认是CBC模式,所以此处都是用的CBC模式

public class DES {

    private DES() {
    }

    /**
     * 加密
     *
     * @param src byte[] 加密的数据源
     * @param password byte[] 加密秘钥
     * @return byte[] 加密后的数据
     */
    public static byte[] encrypt(byte[] src, byte[] password) {
        try {
            DESKeySpec desKey = new DESKeySpec(password);
            //创建一个密匙工厂,然后用它把DESKeySpec转换成
            SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
            SecretKey securekey = keyFactory.generateSecret(desKey);
            //Cipher对象实际完成加密操作
            Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
            //ECB模式下,iv不需要
            IvParameterSpec iv = new IvParameterSpec(password);
            //用密匙初始化Cipher对象
            cipher.init(Cipher.ENCRYPT_MODE, securekey, iv);
            //现在,获取数据并加密
            //正式执行加密操作
            return cipher.doFinal(src);
        } catch (InvalidKeyException | NoSuchAlgorithmException | InvalidKeySpecException | NoSuchPaddingException | IllegalBlockSizeException | BadPaddingException | InvalidAlgorithmParameterException ex) {
            Logger.getLogger(DES.class.getName()).log(Level.SEVERE, null, ex);
        }
        return null;
    }

    /**
     * 加密
     *
     * @param src byte[] 加密的数据源
     * @param password String 加密秘钥
     * @return byte[] 加密后的数据
     */
    public static byte[] encrypt(byte[] src, String password) {
        byte[] keyBytes = password.getBytes(Charset.utf8);
        return encrypt(src, keyBytes);
    }

    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password byte[] 解密秘钥
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, byte[] password) {
        try {
            // DES算法要求有一个可信任的随机数源
            //SecureRandom random = new SecureRandom();
            // 创建一个DESKeySpec对象
            DESKeySpec desKey = new DESKeySpec(password);
            // 创建一个密匙工厂
            SecretKeyFactory keyFactory = SecretKeyFactory.getInstance("DES");
            // 将DESKeySpec对象转换成SecretKey对象
            SecretKey securekey = keyFactory.generateSecret(desKey);
            // Cipher对象实际完成解密操作,CBC为加密模式,
            Cipher cipher = Cipher.getInstance("DES/CBC/PKCS5Padding");
            //ECB模式下,iv不需要
            IvParameterSpec iv = new IvParameterSpec(password);
            // 用密匙初始化Cipher对象
            cipher.init(Cipher.DECRYPT_MODE, securekey, iv);
            // 真正开始解密操作
            return cipher.doFinal(src);
        } catch (InvalidKeyException | NoSuchAlgorithmException | InvalidKeySpecException | NoSuchPaddingException | InvalidAlgorithmParameterException | IllegalBlockSizeException | BadPaddingException ex) {
            Logger.getLogger(DES.class.getName()).log(Level.SEVERE, null, ex);
        }
        return null;
    }
    
    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password String 解密秘钥
     * @param psdCharset 秘钥字符编码
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, String password, java.nio.charset.Charset psdCharset) {
        byte[] keyBytes = password.getBytes(psdCharset);
        return decrypt(src, keyBytes);
    }

    /**
     * 解密
     *
     * @param src byte[] 解密的数据源
     * @param password String 解密秘钥
     * @return byte[] 解密后的数据
     */
    public static byte[] decrypt(byte[] src, String password) {
        return decrypt(src, password, Charset.utf8);
    }
}

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文