博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
AES对称加密,以及javax.crypto.BadPaddingException: Given final block not properly padded 解决
阅读量:4042 次
发布时间:2019-05-24

本文共 4019 字,大约阅读时间需要 13 分钟。

下面的AES加密算法,加密出来的字符串存在“\n”的情况,这影响http中数据传递的特殊字符,解决办法

1.使用Base64再次加密

2.使用 jce(Java Cryptography Extension)的强加密算法,如果再安全保险,加密完后再Base64处理一下

加密代码如下:

public class AESUtil {    /*   * 加密   * 1.构造密钥生成器   * 2.根据ecnodeRules规则初始化密钥生成器   * 3.产生密钥   * 4.创建和初始化密码器   * 5.内容加密   * 6.返回字符串   */    public static String AESEncode(String encodeRules,String content){        String aesEncode = null;        try {            //1.构造密钥生成器,指定为AES算法,不区分大小写            KeyGenerator keygen=KeyGenerator.getInstance("AES");            //2.根据ecnodeRules规则初始化密钥生成器            //生成一个128位的随机源,根据传入的字节数组            SecureRandom secureRandom = SecureRandom.getInstance("SHA1PRNG") ;            secureRandom.setSeed(encodeRules.getBytes());            keygen.init(128, secureRandom);            //3.产生原始对称密钥            SecretKey original_key=keygen.generateKey();            //4.获得原始对称密钥的字节数组            byte [] raw=original_key.getEncoded();            //5.根据字节数组生成AES密钥            SecretKey key=new SecretKeySpec(raw, "AES");            //6.根据指定算法AES自成密码器            Cipher cipher=Cipher.getInstance("AES");            //7.初始化密码器,第一个参数为加密(Encrypt_mode)或者解密解密(Decrypt_mode)操作,第二个参数为使用的KEY            cipher.init(Cipher.ENCRYPT_MODE, key);            //8.获取加密内容的字节数组(这里要设置为utf-8)不然内容中如果有中文和英文混合中文就会解密为乱码            byte [] byte_encode=content.getBytes("utf-8");            //9.根据密码器的初始化方式--加密:将数据加密            byte [] byte_AES=cipher.doFinal(byte_encode);            //10.将加密后的数据转换为字符串            //这里用Base64Encoder中会找不到包            //解决办法:            //在项目的Build path中先移除JRE System Library,再添加库JRE System Library,重新编译后就一切正常了。            aesEncode=new BASE64Encoder().encode(byte_AES);            //11.将字符串返回        } catch (Exception e) {            e.printStackTrace();        }//        log.error("AESEncode error,content:"+content);        //如果有错就返加null        return aesEncode;    }    /*     * 解密     * 解密过程:     * 1.同加密1-4步     * 2.将加密后的字符串反纺成byte[]数组     * 3.将加密内容解密     */    public static String AESDecode(String encodeRules,String content){        String aesDecode = null;        try {            //1.构造密钥生成器,指定为AES算法,不区分大小写            KeyGenerator keygen=KeyGenerator.getInstance("AES");            //2.根据ecnodeRules规则初始化密钥生成器            //生成一个128位的随机源,根据传入的字节数组            SecureRandom secureRandom = SecureRandom.getInstance("SHA1PRNG") ;            secureRandom.setSeed(encodeRules.getBytes());            keygen.init(128, secureRandom);            //3.产生原始对称密钥            SecretKey original_key=keygen.generateKey();            //4.获得原始对称密钥的字节数组            byte [] raw=original_key.getEncoded();            //5.根据字节数组生成AES密钥            SecretKey key=new SecretKeySpec(raw, "AES");            //6.根据指定算法AES自成密码器            Cipher cipher=Cipher.getInstance("AES");            //7.初始化密码器,第一个参数为加密(Encrypt_mode)或者解密(Decrypt_mode)操作,第二个参数为使用的KEY            cipher.init(Cipher.DECRYPT_MODE, key);            //8.将加密并编码后的内容解码成字节数组            byte [] byte_content= new BASE64Decoder().decodeBuffer(content);            /*             * 解密             */            byte [] byte_decode=cipher.doFinal(byte_content);            aesDecode=new String(byte_decode,"utf-8");        } catch (Exception e) {            e.printStackTrace();        }        //如果有错就返加null//        log.error("AESDncode error,content:"+content);        return aesDecode;    }    public static void main(String args[]){        String ss = AESUtil.AESEncode("APP_LOGIN",6000875+","+"aadfsdf111"+","+12313);        System.out.print("1:"+ss);        String deS = AESUtil.AESDecode("APP_LOGIN",ss);        System.out.print("2:"+deS);    }}

javax.crypto.BadPaddingException: Given final block not properly padded错误是因为上述红色部分之前代码是

keygen.init(128, new SecureRandom(encodeRules.getBytes()));

原因:SecureRandom 实现完全随操作系统本身的內部状态,除非调用方在 getInstance 方法,然后调用 setSeed 方法;该实现在 windows 上每次生成的 key 都相同,但是在 solaris 或部分 linux 系统上则不同。关于SecureRandom类的详细介绍,见 

解决办法:调用getInstance方法

SecureRandom secureRandom = SecureRandom.getInstance("SHA1PRNG") ;secureRandom.setSeed(encodeRules.getBytes());

 参考地址:

转载地址:http://leadi.baihongyu.com/

你可能感兴趣的文章
git 常用命令
查看>>
linux位操作API
查看>>
uboot.lds文件分析
查看>>
uboot start.s文件分析
查看>>
没有路由器的情况下,开发板,虚拟机Ubuntu,win10主机,三者也可以ping通
查看>>
本地服务方式搭建etcd集群
查看>>
安装k8s Master高可用集群
查看>>
忽略图片透明区域的事件(Flex)
查看>>
忽略图片透明区域的事件(Flex)
查看>>
AS3 Flex基础知识100条
查看>>
Flex动态获取flash资源库文件
查看>>
Flex 中的元数据标签
查看>>
flex4 中创建自定义弹出窗口
查看>>
01Java基础语法-13. if分支语句的灵活使用
查看>>
01Java基础语法-15.for循环结构
查看>>
01Java基础语法-16. while循环结构
查看>>
01Java基础语法-17. do..while循环结构
查看>>
01Java基础语法-18. 各种循环语句的区别和应用场景
查看>>
01Java基础语法-19. 循环跳转控制语句
查看>>
Django框架全面讲解 -- Form
查看>>