Java使用AES加密和解密的实例详解
发布时间 - 2026-01-11 02:18:07 点击率:次Java使用AES加密和解密的实例详解

前言:
AES的基本要求是,采用对称分组密码体制,密钥长度的最少支持为128、192、256,分组长度128位,算法应易于各种硬件和软件实现。1998年NIST开始AES第一轮分析、测试和征集,共产生了15个候选算法。1999年3月完成了第二轮AES2的分析、测试。2000年10月2日美国政府正式宣布选中比利时密码学家Joan Daemen 和 Vincent Rijmen 提出的一种密码算法RIJNDAEL 作为 AES. 在应用方面,尽管DES在安全上是脆弱的,但由于快速DES芯片的大量生产,使得DES仍能暂时继续使用,为提高安全强度,通常使用独立密钥的三级DES。但是DES迟早要被AES代替。流密码体制较之分组密码在理论上成熟且安全,但未被列入下一代加密标准。
AES加密数据块和密钥长度可以是128比特、192比特、256比特中的任意一个。
AES加密有很多轮的重复和变换。大致步骤如下:
1、密钥扩展(KeyExpansion),
2、初始轮(Initial Round),
3、重复轮(Rounds),每一轮又包括:SubBytes、ShiftRows、MixColumns、AddRoundKey,
4、最终轮(Final Round),最终轮没有MixColumns。
AES是一种对称的加密算法,可基于相同的密钥进行加密和解密。Java采用AES算法进行加解密的逻辑大致如下:
1、生成/获取密钥
2、加/解密
1.1生成密钥
密钥的生成是通过KeyGenerator来生成的。通过获取一个KeyGenerator实例,然后调用其generateKey()方法即可生成一个SecretKey对象。大致逻辑一般如下:
private SecretKey geneKey() throws Exception {
//获取一个密钥生成器实例
KeyGenerator keyGenerator = KeyGenerator.getInstance(ALGORITHM);
SecureRandom random = new SecureRandom();
random.setSeed("123456".getBytes());//设置加密用的种子,密钥
keyGenerator.init(random);
SecretKey secretKey = keyGenerator.generateKey();
return secretKey;
}
上述生成密钥的过程中指定了固定的种子,每次生成出来的密钥都是一样的。还有一种形式,我们可以通过不指定SecureRandom对象的种子,即不调用其setSeed方法,这样每次生成出来的密钥都可能是不一样的。
private SecretKey geneKey() throws Exception {
//获取一个密钥生成器实例
KeyGenerator keyGenerator = KeyGenerator.getInstance(ALGORITHM);
SecureRandom random = new SecureRandom();
keyGenerator.init(random);
SecretKey secretKey = keyGenerator.generateKey();
return secretKey;
}
通过KeyGenerator的init(keySize)方法进行初始化,而不是通过传递SecureRandom对象进行初始化也可以达到上面的效果,每次生成的密钥都可能是不一样的。但是对应的keySize的指定一定要正确,AES算法的keySize是128。
private SecretKey geneKey() throws Exception {
//获取一个密钥生成器实例
KeyGenerator keyGenerator = KeyGenerator.getInstance(ALGORITHM);
keyGenerator.init(128);
SecretKey secretKey = keyGenerator.generateKey();
return secretKey;
}
但是这种每次生成出来的密钥都是不同的情况下,我们需要把加密用的密钥存储起来,以供解密的时候使用,不然就没法进行解密了。
1.2密钥的存储
密钥SecretKey里面最核心的内容就是其中的密钥对应的字节数组,可以通过SecretKey的getEncoded()方法获取。然后把它存储起来即可。最简单的方式就是直接写入一个文件中。
//把上面的密钥存起来
Path keyPath = Paths.get("D:/aes.key");
Files.write(keyPath, secretKey.getEncoded());
1.3获取存储的密钥
获取存储的密钥的核心是把密钥的字节数组转换为对应的SecretKey。这可以通过SecretKeySpec来获取,其实现了SecretKey接口,然后构造参数里面将接收密钥的字节数组。
private SecretKey readKey(Path keyPath) throws Exception {
//读取存起来的密钥
byte[] keyBytes = Files.readAllBytes(keyPath);
SecretKeySpec keySpec = new SecretKeySpec(keyBytes, ALGORITHM);
return keySpec;
}
1.4加解密
Java采用AES算法进行加解密的过程是类似的,具体如下:
1、指定算法,获取一个Cipher实例对象
Cipher cipher = Cipher.getInstance(ALGORITHM);//算法是AES
2、生成/读取用于加解密的密钥
SecretKey secretKey = this.geneKey();
3、用指定的密钥初始化Cipher对象,同时指定加解密模式,是加密模式还是解密模式。
cipher.init(Cipher.ENCRYPT_MODE, secretKey);
4、通过update指定需要加密的内容,不可多次调用。
cipher.update(content.getBytes());
5、通过Cipher的dofinal()进行最终的加解密操作。
byte[] result = cipher.doFinal();//加密后的字节数组
通过以上几步就完成了使用AES算法进行加解密的操作了。其实第4、5步是可以合在一起的,即在进行doFinal的时候传递需要进行加解密的内容。但是如果update指定了加密的内容,而doFinal的时候也指定了加密的内容,那最终加密出来的结果将是两次指定的加密内容的和对应的加密结果。
byte[] result = cipher.doFinal(content.getBytes());
以下是一次加解密操作的完整示例。
public class AESTest {
private static final String ALGORITHM = "AES";
/**
* 生成密钥
* @return
* @throws Exception
*/
private SecretKey geneKey() throws Exception {
//获取一个密钥生成器实例
KeyGenerator keyGenerator = KeyGenerator.getInstance(ALGORITHM);
SecureRandom random = new SecureRandom();
random.setSeed("123456".getBytes());//设置加密用的种子,密钥
keyGenerator.init(random);
SecretKey secretKey = keyGenerator.generateKey();
//把上面的密钥存起来
Path keyPath = Paths.get("D:/aes.key");
Files.write(keyPath, secretKey.getEncoded());
return secretKey;
}
/**
* 读取存储的密钥
* @param keyPath
* @return
* @throws Exception
*/
private SecretKey readKey(Path keyPath) throws Exception {
//读取存起来的密钥
byte[] keyBytes = Files.readAllBytes(keyPath);
SecretKeySpec keySpec = new SecretKeySpec(keyBytes, ALGORITHM);
return keySpec;
}
/**
* 加密测试
*/
@Test
public void testEncrypt() throws Exception {
//1、指定算法、获取Cipher对象
Cipher cipher = Cipher.getInstance(ALGORITHM);//算法是AES
//2、生成/读取用于加解密的密钥
SecretKey secretKey = this.geneKey();
//3、用指定的密钥初始化Cipher对象,指定是加密模式,还是解密模式
cipher.init(Cipher.ENCRYPT_MODE, secretKey);
String content = "Hello AES";//需要加密的内容
//4、更新需要加密的内容
cipher.update(content.getBytes());
//5、进行最终的加解密操作
byte[] result = cipher.doFinal();//加密后的字节数组
//也可以把4、5步组合到一起,但是如果保留了4步,同时又是如下这样使用的话,加密的内容将是之前update传递的内容和doFinal传递的内容的和。
// byte[] result = cipher.doFinal(content.getBytes());
String base64Result = Base64.getEncoder().encodeToString(result);//对加密后的字节数组进行Base64编码
System.out.println("Result: " + base64Result);
}
/**
* 解密测试
*/
@Test
public void testDecrpyt() throws Exception {
Cipher cipher = Cipher.getInstance(ALGORITHM);
SecretKey secretKey = this.geneKey();
cipher.init(Cipher.DECRYPT_MODE, secretKey);
String content = "pK9Xw4zqTMXYraDadSGJE3x/ftrDxIg2AM/acq0uixA=";//经过Base64加密的待解密的内容
byte[] encodedBytes = Base64.getDecoder().decode(content.getBytes());
byte[] result = cipher.doFinal(encodedBytes);//对加密后的字节数组进行解密
System.out.println("Result: " + new String(result));
}
}
1.5使用存储的密钥进行加解密示例
@Test
public void test() throws Exception {
Cipher cipher = Cipher.getInstance(ALGORITHM);
KeyGenerator keyGenerator = KeyGenerator.getInstance(ALGORITHM);
keyGenerator.init(128);
SecretKey secretKey = keyGenerator.generateKey();
//把上面的密钥存起来
Path keyPath = Paths.get("D:/aes.key");
Files.write(keyPath, secretKey.getEncoded());
//读取存起来的密钥
SecretKey key = this.readKey(keyPath);
cipher.init(Cipher.ENCRYPT_MODE, key);
cipher.update("Hello World".getBytes());
//密文
byte[] encryptBytes = cipher.doFinal();
System.out.println(Base64.getEncoder().encodeToString(encryptBytes));
//用取出来的密钥进行解密
cipher.init(Cipher.DECRYPT_MODE, key);
//明文
byte[] decryptBytes = cipher.doFinal(encryptBytes);
System.out.println(new String(decryptBytes));
}
在上面的示例中,我们先生成了一个密钥,然后把它保存到本地文件中,然后再把它读出来,分别用以加密和解密。而且我们加密和解密都是用的同一个Cipher对象,但是在使用前需要重新通过init方法初始化加解密模式。
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!
# Java使用AES加密和解密
# java
# AES加解密
# Java AES加密和解密教程
# Java实现AES加密和解密方式完整示例
# Java AES加密解密的简单实现方法
# JAVA中AES对称加密和解密过程
# 使用Java实现加密之AES加解密
# Java中使用Hutool进行AES加密解密的方法举例
# java AES加密/解密实现完整代码(附带源码)
# 加解密
# 把它
# 都是
# 将是
# 国政
# 日美
# 比利时
# 完成了
# 是一种
# 成了
# 又是
# 有很多
# 两次
# 也可
# 我们可以
# 作了
# 可以通过
# 希望能
# 然后再
# 在上面
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
装修招标网站设计制作流程,装修招标流程?
android nfc常用标签读取总结
EditPlus中的正则表达式 实战(4)
Laravel如何处理CORS跨域请求?(配置示例)
javascript读取文本节点方法小结
电商网站制作多少钱一个,电子商务公司的网站制作费用计入什么科目?
Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程
javascript中的数组方法有哪些_如何利用数组方法简化数据处理
Laravel用户密码怎么加密_Laravel Hash门面使用教程
如何在阿里云完成域名注册与建站?
微信小程序 input输入框控件详解及实例(多种示例)
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤
html5audio标签播放结束怎么触发事件_onended回调方法【教程】
EditPlus中的正则表达式 实战(1)
今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】
如何用y主机助手快速搭建网站?
Android自定义控件实现温度旋转按钮效果
电视网站制作tvbox接口,云海电视怎样自定义添加电视源?
高端智能建站公司优选:品牌定制与SEO优化一站式服务
mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?
LinuxCD持续部署教程_自动发布与回滚机制
微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)
laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析
PHP正则匹配日期和时间(时间戳转换)的实例代码
Win11怎么设置虚拟桌面 Win11新建多桌面切换操作【技巧】
Win11怎么恢复误删照片_Win11数据恢复工具使用【推荐】
Laravel怎么使用Blade模板引擎_Laravel模板继承与Component组件复用【手册】
如何使用 jQuery 正确渲染 Instagram 风格的标签列表
JavaScript模板引擎Template.js使用详解
如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】
Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】
如何利用DOS批处理实现定时关机操作详解
JS中页面与页面之间超链接跳转中文乱码问题的解决办法
动图在线制作网站有哪些,滑动动图图集怎么做?
非常酷的网站设计制作软件,酷培ai教育官方网站?
如何在 Pandas 中基于一列条件计算另一列的分组均值
猎豹浏览器开发者工具怎么打开 猎豹浏览器F12调试工具使用【前端必备】
Laravel任务队列怎么用_Laravel Queues异步处理任务提升应用性能
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?
Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程
百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏
悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】
Python结构化数据采集_字段抽取解析【教程】
西安专业网站制作公司有哪些,陕西省建行官方网站?
米侠浏览器网页图片不显示怎么办 米侠图片加载修复
大同网页,大同瑞慈医院官网?
如何快速上传自定义模板至建站之星?

