RSA加密内容过长导致抛异常javax.crypto.IllegalBlockSizeException: Data must not be longer than 117 bytes

本文主要是介绍RSA加密内容过长导致抛异常javax.crypto.IllegalBlockSizeException: Data must not be longer than 117 bytes,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在进行对内容加密时发现内容长度过长就会导致Data must not be longer than 117
bytes的问题,后来百度的一番才得知由于是RSA的加密长度只能加密117bytes的内容,所以就报错了。


搜索网上的内容给出的解决办法有两种,一种是分段进行加密数据,另一种加密是使用RSA加密AES的密码,然后使用AES来加密数据。两者比较推荐使用后者,前者的加密方式确实安全,每次加密解密太慢了,如果使用AES加密速度就快了很多,但是不足够安全,这时候使用RSA来加密一下AES的秘钥,这样就更加的安全了。

RSA加密方案逻辑图

实现示例:

RSA工具类

package com.bzqll.util;import com.sun.org.apache.xml.internal.security.exceptions.Base64DecodingException;
import com.sun.org.apache.xml.internal.security.utils.Base64;
import com.sun.xml.internal.rngom.parse.host.Base;
import org.apache.commons.io.output.ByteArrayOutputStream;
import org.springframework.util.Base64Utils;import javax.crypto.BadPaddingException;
import javax.crypto.Cipher;
import javax.crypto.IllegalBlockSizeException;
import java.io.IOException;
import java.security.*;
import java.security.interfaces.RSAPrivateKey;
import java.security.interfaces.RSAPublicKey;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;public class RSAHelper {public static final String PUBLIC_KEY_VALUES = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA0bb7yxuvRIWOUxlTbTXE\n" +"cKX5f4Q6+BBOf1fYJKic9l6Wf1QPmyt6ML7PywaPH861D7eYoQl0bGNK2fKsgcAG\n" +"ZzObG5CpmP8ESnSzqcjltdAgx+neCZQy7yUmXUIhpBEQMN80CNYoasOxeZTdPh2w\n" +"zhlmwa27ubkvpINtKUfZbg8sQ5wiDGbLM32ej8z2Rl8DNY4vrusJaNXB7LWaRQm7\n" +"4lPhLN2B/hMv/Ktif4iNxUCYDY97Xws2kVVu7ffWkn4rnhiCrTw2XMZRjIJq3a4o\n" +"4zGWUhYm0usVOLz+yG22cLSCIDhM8tBXL2f3960l4OIQbSObfQkGnqlmi0Fe686p\n" +"gwIDAQAB";public static final String PRIVATE_KEY_VALUES = "MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQDRtvvLG69EhY5T\n" +"GVNtNcRwpfl/hDr4EE5/V9gkqJz2XpZ/VA+bK3owvs/LBo8fzrUPt5ihCXRsY0rZ\n" +"8qyBwAZnM5sbkKmY/wRKdLOpyOW10CDH6d4JlDLvJSZdQiGkERAw3zQI1ihqw7F5\n" +"lN0+HbDOGWbBrbu5uS+kg20pR9luDyxDnCIMZsszfZ6PzPZGXwM1ji+u6wlo1cHs\n" +"tZpFCbviU+Es3YH+Ey/8q2J/iI3FQJgNj3tfCzaRVW7t99aSfiueGIKtPDZcxlGM\n" +"gmrdrijjMZZSFibS6xU4vP7IbbZwtIIgOEzy0FcvZ/f3rSXg4hBtI5t9CQaeqWaL\n" +"QV7rzqmDAgMBAAECggEBAIpVZ5Y8tso+RvmttQhO9TtRYFirArLrHryFV4Am8RLa\n" +"pe9rpbe3YCyTUUIdG3/hvDGX68geEnwEqzVFDGGyJwcgVWpDbHwNi+kJPhH7APuD\n" +"RHxaWip7ZXW2Ta4ql0JANyvlr888ZQC4AXOvrJjywNPSkaOkVDZYX4LnZrRaPqSH\n" +"j9iVGuKK4nbqbXP1tuLyto6bsn3L2KuwrBthtiDPaZ1cRm3aSu/XND030iSGloW/\n" +"/f6vHI3U//cPFPdi8p54nbpGTB0E66D2xbfK5+z9dZYoFDZUdMsloR1JgS4vQErr\n" +"17165RS7cpWx2HejiKat6qmNFXkHbREpcLDT+lq91LECgYEA7wqm+mIFIYyXEARc\n" +"lKKabCKaufKE+7kyBGpAEyon1lYZHw62dCYkGmuXULam4IAz7TesWsbG5SEgDx/A\n" +"jeFSoC6NTyyTzSh1I08UHdOJ4xhAa67GnlJG7WLNZFK+itl7v3VOOlGsAYAxaG5q\n" +"kQKQ1mKk6C6oGRpQ0aqKxVhoDc0CgYEA4Je2flPUYs33qy4m09UBFx6Xf91/5LvL\n" +"iTtEo+HGcMzxmfeV3sawUwtjHhyippN5lTR80JLbYoEvoUMyKt+N5Jnic206alSH\n" +"hujsQnO4LCRHz/XwkHnB8Ncd3J67Uaq9PUPY630NIA/ChBAQchsqntJSdVn65PZf\n" +"mw7+hO2WxI8CgYB4eX/qlVhMrlS8R9Z5OvJlKZOdv/LyA0aIHxyoDAkD52TF2F5w\n" +"b4CmqC8dCNFWOIbzOanuHlzDwkwsEy6y0ysXfB7QFoFvVsKixwo2dhT6lZByNSX5\n" +"STJiFfe6ZlGOHUpXFkIU9nCgWQGNxoiDCS4CPrkqI8mozTCKW0+RYpseyQKBgQCR\n" +"cK21UQQQl+Dy8YgjVaTHHABvxTi1HwfHbqIcnnCrS4yJcFOVWIWGwbEGJvUNeiMa\n" +"BEtvpip7t7zoaWNrcCmrCBwlM27IvMSnEN8uiVGTBEuc2F9YsABvvl6QKBqV4EN8\n" +"ERvAI9MEGDCW5PBBdGY9Q2YyqHpZG1L+Ts9ztYgU8QKBgB7bP9eNMrDzgZ6i9pFP\n" +"djlcJ+NE0veOLwTA1eQhiIZFSJTI5eyiePTYNIS3GwIyGauzePw5LeBb6rLVBt4Y\n" +"NAOgJDeHbPV2bnstONjE7FyUswJivIVD3n3UaVGgBTe6xc468Ws4rmeKZ8/Ph1Nq\n" +"ylaViyPWz486JAibF3Kudl5B";/*** RSA密钥长度必须是64的倍数,在512~65536之间。默认是1024*/public static final int KEY_SIZE = 2048;/*** 生成公钥、私钥对(keysize=1024)*/public static RSAHelper.KeyPairInfo getKeyPair() {return getKeyPair(KEY_SIZE);}/*** 生成公钥、私钥对** @param keySize* @return*/public static RSAHelper.KeyPairInfo getKeyPair(int keySize) {try {KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA");keyPairGen.initialize(keySize);// 生成一个密钥对,保存在keyPair中KeyPair keyPair = keyPairGen.generateKeyPair();// 得到私钥RSAPrivateKey oraprivateKey = (RSAPrivateKey) keyPair.getPrivate();// 得到公钥RSAPublicKey orapublicKey = (RSAPublicKey) keyPair.getPublic();RSAHelper.KeyPairInfo pairInfo = new RSAHelper.KeyPairInfo(keySize);//公钥byte[] publicKeybyte = orapublicKey.getEncoded();String publicKeyString = Base64.encode(publicKeybyte);pairInfo.setPublicKey(publicKeyString);//私钥byte[] privateKeybyte = oraprivateKey.getEncoded();String privateKeyString = Base64.encode(privateKeybyte);pairInfo.setPrivateKey(privateKeyString);return pairInfo;} catch (Exception e) {e.printStackTrace();return null;}}/*** 获取公钥对象** @param publicKeyBase64* @return* @throws InvalidKeySpecException* @throws NoSuchAlgorithmException*/public static PublicKey getPublicKey(String publicKeyBase64)throws InvalidKeySpecException, NoSuchAlgorithmException, Base64DecodingException {KeyFactory keyFactory = KeyFactory.getInstance("RSA");X509EncodedKeySpec publicpkcs8KeySpec =new X509EncodedKeySpec(Base64.decode(publicKeyBase64));PublicKey publicKey = keyFactory.generatePublic(publicpkcs8KeySpec);return publicKey;}/*** 获取私钥对象** @param privateKeyBase64* @return* @throws NoSuchAlgorithmException* @throws InvalidKeySpecException*/public static PrivateKey getPrivateKey(String privateKeyBase64)throws NoSuchAlgorithmException, InvalidKeySpecException, Base64DecodingException {KeyFactory keyFactory = KeyFactory.getInstance("RSA");PKCS8EncodedKeySpec privatekcs8KeySpec =new PKCS8EncodedKeySpec(Base64.decode(privateKeyBase64));PrivateKey privateKey = keyFactory.generatePrivate(privatekcs8KeySpec);return privateKey;}/*** 使用工钥加密** @param content         待加密内容* @param publicKeyBase64 公钥 base64 编码* @return 经过 base64 编码后的字符串*/public static String encipher(String content, String publicKeyBase64) {return encipher(content, publicKeyBase64, KEY_SIZE / 8 - 11);}/*** 使用公司钥加密(分段加密)** @param content         待加密内容* @param publicKeyBase64 公钥 base64 编码* @param segmentSize     分段大小,一般小于 keySize/8(段小于等于0时,将不使用分段加密)* @return 经过 base64 编码后的字符串*/public static String encipher(String content, String publicKeyBase64, int segmentSize) {try {PublicKey publicKey = getPublicKey(publicKeyBase64);return encipher(content, publicKey, segmentSize);} catch (Exception e) {e.printStackTrace();return null;}}/*** 分段加密** @param ciphertext  密文* @param key         加密秘钥* @param segmentSize 分段大小,<=0 不分段* @return*/public static String encipher(String ciphertext, java.security.Key key, int segmentSize) {try {// 用公钥加密byte[] srcBytes = ciphertext.getBytes();// Cipher负责完成加密或解密工作,基于RSACipher cipher = Cipher.getInstance("RSA");// 根据公钥,对Cipher对象进行初始化cipher.init(Cipher.ENCRYPT_MODE, key);byte[] resultBytes = null;if (segmentSize > 0)resultBytes = cipherDoFinal(cipher, srcBytes, segmentSize); //分段加密elseresultBytes = cipher.doFinal(srcBytes);String base64Str = Base64Utils.encodeToString(resultBytes);return base64Str;} catch (Exception e) {e.printStackTrace();return null;}}/*** 分段大小** @param cipher* @param srcBytes* @param segmentSize* @return* @throws IllegalBlockSizeException* @throws BadPaddingException* @throws IOException*/public static byte[] cipherDoFinal(Cipher cipher, byte[] srcBytes, int segmentSize)throws IllegalBlockSizeException, BadPaddingException, IOException {if (segmentSize <= 0)throw new RuntimeException("分段大小必须大于0");ByteArrayOutputStream out = new ByteArrayOutputStream();int inputLen = srcBytes.length;int offSet = 0;byte[] cache;int i = 0;// 对数据分段解密while (inputLen - offSet > 0) {if (inputLen - offSet > segmentSize) {cache = cipher.doFinal(srcBytes, offSet, segmentSize);} else {cache = cipher.doFinal(srcBytes, offSet, inputLen - offSet);}out.write(cache, 0, cache.length);i++;offSet = i * segmentSize;}byte[] data = out.toByteArray();out.close();return data;}/*** 使用私钥解密** @param contentBase64    待加密内容,base64 编码* @param privateKeyBase64 私钥 base64 编码* @return* @segmentSize 分段大小*/public static String decipher(String contentBase64, String privateKeyBase64) {return decipher(contentBase64, privateKeyBase64, KEY_SIZE / 8);}/*** 使用私钥解密(分段解密)** @param contentBase64    待加密内容,base64 编码* @param privateKeyBase64 私钥 base64 编码* @return* @segmentSize 分段大小*/public static String decipher(String contentBase64, String privateKeyBase64, int segmentSize) {try {PrivateKey privateKey = getPrivateKey(privateKeyBase64);return decipher(contentBase64, privateKey, segmentSize);} catch (Exception e) {e.printStackTrace();return null;}}/*** 分段解密** @param contentBase64 密文* @param key           解密秘钥* @param segmentSize   分段大小(小于等于0不分段)* @return*/public static String decipher(String contentBase64, java.security.Key key, int segmentSize) {try {// 用私钥解密byte[] srcBytes = Base64Utils.decodeFromString(contentBase64);// Cipher负责完成加密或解密工作,基于RSACipher deCipher = Cipher.getInstance("RSA");// 根据公钥,对Cipher对象进行初始化deCipher.init(Cipher.DECRYPT_MODE, key);byte[] decBytes = null;//deCipher.doFinal(srcBytes);if (segmentSize > 0)decBytes = cipherDoFinal(deCipher, srcBytes, segmentSize); //分段加密elsedecBytes = deCipher.doFinal(srcBytes);String decrytStr = new String(decBytes);return decrytStr;} catch (Exception e) {e.printStackTrace();return null;}}/*** 秘钥对*/public static class KeyPairInfo {String privateKey;String publicKey;int keySize = 0;public KeyPairInfo(int keySize) {setKeySize(keySize);}public KeyPairInfo(String publicKey, String privateKey) {setPrivateKey(privateKey);setPublicKey(publicKey);}public String getPrivateKey() {return privateKey;}public void setPrivateKey(String privateKey) {this.privateKey = privateKey;}public String getPublicKey() {return publicKey;}public void setPublicKey(String publicKey) {this.publicKey = publicKey;}public int getKeySize() {return keySize;}public void setKeySize(int keySize) {this.keySize = keySize;}}
}

AES工具类

package com.bzqll.util;import org.springframework.util.Base64Utils;import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.crypto.Cipher;
import javax.crypto.KeyGenerator;
import javax.crypto.SecretKey;
import javax.crypto.spec.SecretKeySpec;/*** @version V1.0* @desc AES 加密工具类*/
public class AESUtil {private static final String KEY_ALGORITHM = "AES";private static final String DEFAULT_CIPHER_ALGORITHM = "AES/ECB/PKCS5Padding";//默认的加密算法/*** AES 加密操作** @param content  待加密内容* @param password 加密密码* @return 返回Base64转码后的加密数据*/public static String encrypt(String content, String password) {try {Cipher cipher = Cipher.getInstance(DEFAULT_CIPHER_ALGORITHM);// 创建密码器byte[] byteContent = content.getBytes("utf-8");cipher.init(Cipher.ENCRYPT_MODE, getSecretKey(password));// 初始化为加密模式的密码器byte[] result = cipher.doFinal(byteContent);// 加密return Base64Utils.encodeToString(result);//通过Base64转码返回} catch (Exception ex) {Logger.getLogger(AESUtil.class.getName()).log(Level.SEVERE, null, ex);}return null;}/*** AES 解密操作** @param content* @param password* @return*/public static String decrypt(String content, String password) {try {//实例化Cipher cipher = Cipher.getInstance(DEFAULT_CIPHER_ALGORITHM);//使用密钥初始化,设置为解密模式cipher.init(Cipher.DECRYPT_MODE, getSecretKey(password));//执行操作byte[] result = cipher.doFinal(Base64Utils.decodeFromString(content));return new String(result, "utf-8");} catch (Exception ex) {Logger.getLogger(AESUtil.class.getName()).log(Level.SEVERE, null, ex);}return null;}/*** 生成加密秘钥** @return*/private static SecretKeySpec getSecretKey(String password) {//返回生成指定算法密钥生成器的 KeyGenerator 对象KeyGenerator kg = null;try {kg = KeyGenerator.getInstance(KEY_ALGORITHM);//AES 要求密钥长度为 128kg.init(128, new SecureRandom(password.getBytes()));//生成一个密钥SecretKey secretKey = kg.generateKey();return new SecretKeySpec(secretKey.getEncoded(), KEY_ALGORITHM);// 转换为AES专用密钥} catch (NoSuchAlgorithmException ex) {Logger.getLogger(AESUtil.class.getName()).log(Level.SEVERE, null, ex);}return null;}}

由于是公司业务代码,实际代码中的逻辑更复杂,后面的逻辑类就不贴出来了,只贴出了工具类

这篇关于RSA加密内容过长导致抛异常javax.crypto.IllegalBlockSizeException: Data must not be longer than 117 bytes的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/520552

相关文章

在Ubuntu上部署SpringBoot应用的操作步骤

《在Ubuntu上部署SpringBoot应用的操作步骤》随着云计算和容器化技术的普及,Linux服务器已成为部署Web应用程序的主流平台之一,Java作为一种跨平台的编程语言,具有广泛的应用场景,本... 目录一、部署准备二、安装 Java 环境1. 安装 JDK2. 验证 Java 安装三、安装 mys

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

SpringCloud集成AlloyDB的示例代码

《SpringCloud集成AlloyDB的示例代码》AlloyDB是GoogleCloud提供的一种高度可扩展、强性能的关系型数据库服务,它兼容PostgreSQL,并提供了更快的查询性能... 目录1.AlloyDBjavascript是什么?AlloyDB 的工作原理2.搭建测试环境3.代码工程1.

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

SpringBoot操作spark处理hdfs文件的操作方法

《SpringBoot操作spark处理hdfs文件的操作方法》本文介绍了如何使用SpringBoot操作Spark处理HDFS文件,包括导入依赖、配置Spark信息、编写Controller和Ser... 目录SpringBoot操作spark处理hdfs文件1、导入依赖2、配置spark信息3、cont

springboot整合 xxl-job及使用步骤

《springboot整合xxl-job及使用步骤》XXL-JOB是一个分布式任务调度平台,用于解决分布式系统中的任务调度和管理问题,文章详细介绍了XXL-JOB的架构,包括调度中心、执行器和Web... 目录一、xxl-job是什么二、使用步骤1. 下载并运行管理端代码2. 访问管理页面,确认是否启动成功

Java中的密码加密方式

《Java中的密码加密方式》文章介绍了Java中使用MD5算法对密码进行加密的方法,以及如何通过加盐和多重加密来提高密码的安全性,MD5是一种不可逆的哈希算法,适合用于存储密码,因为其输出的摘要长度固... 目录Java的密码加密方式密码加密一般的应用方式是总结Java的密码加密方式密码加密【这里采用的

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(

解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题

《解决mybatis-plus-boot-starter与mybatis-spring-boot-starter的错误问题》本文主要讲述了在使用MyBatis和MyBatis-Plus时遇到的绑定异常... 目录myBATis-plus-boot-starpythonter与mybatis-spring-b