Java实现RSA加密工具类
作者:崔笑颜 发布时间:2023-08-23 08:41:02
标签:Java,rsa,加密,工具类
公钥加密算法,也就是 非对称加密算法,这种算法加密和解密的密码不一样,一个是公钥,另一个是私钥:
公钥和私钥成对出现
公开的密钥叫公钥,只有自己知道的叫私钥
用公钥加密的数据只有对应的私钥可以解密
用私钥加密的数据只有对应的公钥可以解密
如果可以用公钥解密,则必然是对应的私钥加的密
如果可以用私钥解密,则必然是对应的公钥加的密
公钥和私钥是相对的,两者本身并没有规定哪一个必须是公钥或私钥。
代码如下
package com.cxy.template.controller.keyTools;
import java.util.Base64;
import javax.crypto.Cipher;
import java.security.KeyFactory;
import java.security.KeyPair;
import java.security.KeyPairGenerator;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.security.interfaces.RSAPrivateKey;
import java.security.interfaces.RSAPublicKey;
import java.security.spec.PKCS8EncodedKeySpec;
import java.security.spec.X509EncodedKeySpec;
import java.util.HashMap;
import java.util.Map;
/**
* Java RSA 加密工具类
*
*/
public class RSAUtils {
/**
* 密钥长度 于原文长度对应 以及越长速度越慢
*/
private final static int KEY_SIZE = 1024;
/**
* 用于封装随机产生的公钥与私钥
*/
private static Map<Integer, String> keyMap = new HashMap<Integer, String>();
/**
* 随机生成密钥对
*/
public static void genKeyPair() throws NoSuchAlgorithmException {
// KeyPairGenerator类用于生成公钥和私钥对,基于RSA算法生成对象
KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA");
// 初始化密钥对生成器
keyPairGen.initialize(KEY_SIZE, new SecureRandom());
// 生成一个密钥对,保存在keyPair中
KeyPair keyPair = keyPairGen.generateKeyPair();
// 得到私钥
RSAPrivateKey privateKey = (RSAPrivateKey) keyPair.getPrivate();
// 得到公钥
RSAPublicKey publicKey = (RSAPublicKey) keyPair.getPublic();
String publicKeyString = Base64.getEncoder().encodeToString(publicKey.getEncoded());
// 得到私钥字符串
String privateKeyString = Base64.getEncoder().encodeToString(privateKey.getEncoded());
// 将公钥和私钥保存到Map
//0表示公钥
keyMap.put(0, publicKeyString);
//1表示私钥
keyMap.put(1, privateKeyString);
}
/**
* RSA公钥加密
*
* @param str 加密字符串
* @param publicKey 公钥
* @return 密文
* @throws Exception 加密过程中的异常信息
*/
public static String encrypt(String str, String publicKey) throws Exception {
//base64编码的公钥
byte[] decoded = Base64.getDecoder().decode(publicKey);
RSAPublicKey pubKey = (RSAPublicKey) KeyFactory.getInstance("RSA").generatePublic(new X509EncodedKeySpec(decoded));
//RSA加密
Cipher cipher = Cipher.getInstance("RSA");
cipher.init(Cipher.ENCRYPT_MODE, pubKey);
String outStr = Base64.getEncoder().encodeToString(cipher.doFinal(str.getBytes("UTF-8")));
return outStr;
}
/**
* RSA私钥解密
*
* @param str 加密字符串
* @param privateKey 私钥
* @return 明文
* @throws Exception 解密过程中的异常信息
*/
public static String decrypt(String str, String privateKey) throws Exception {
//64位解码加密后的字符串
byte[] inputByte = Base64.getDecoder().decode(str);
//base64编码的私钥
byte[] decoded = Base64.getDecoder().decode(privateKey);
RSAPrivateKey priKey = (RSAPrivateKey) KeyFactory.getInstance("RSA").generatePrivate(new PKCS8EncodedKeySpec(decoded));
//RSA解密
Cipher cipher = Cipher.getInstance("RSA");
cipher.init(Cipher.DECRYPT_MODE, priKey);
String outStr = new String(cipher.doFinal(inputByte));
return outStr;
}
public static void main(String[] args) throws Exception {
long temp = System.currentTimeMillis();
//生成公钥和私钥
genKeyPair();
//加密字符串
System.out.println("公钥:" + keyMap.get(0));
System.out.println("私钥:" + keyMap.get(1));
System.out.println("生成密钥消耗时间:" + (System.currentTimeMillis() - temp) / 1000.0 + "秒");
//客户id + 授权时间 + 所用模块
String message = "4028138151b3cf300151b419df090007" + "2015-12-17 11:30:22" + "A01,A02";
System.out.println("原文:" + message);
temp = System.currentTimeMillis();
//通过原文,和公钥加密。
String messageEn = encrypt(message, keyMap.get(0));
System.out.println("密文:" + messageEn);
System.out.println("加密消耗时间:" + (System.currentTimeMillis() - temp) / 1000.0 + "秒");
temp = System.currentTimeMillis();
//通过密文,和私钥解密。
String messageDe = decrypt(messageEn, keyMap.get(1));
System.out.println("解密:" + messageDe);
System.out.println("解密消耗时间:" + (System.currentTimeMillis() - temp) / 1000.0 + "秒");
}
}
测试如下
公钥:MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCtI8/X2ayQNXsz6/HTJew8z86+aevsJJLNS5dT74/zUwfaOpouInMQLwhqLFvMoCIzw2CLdG7/hwx3HGlKTHDpaTCqnKu17NVOriI8o7lAgI6pK/L2M8kpkbAG88XBmH6aNbi94TS1/ljsS9jMNoEQvfLC2A/KUw9sopRg73gAawIDAQAB
私钥:MIICdwIBADANBgkqhkiG9w0BAQEFAASCAmEwggJdAgEAAoGBAK0jz9fZrJA1ezPr8dMl7DzPzr5p6+wkks1Ll1Pvj/NTB9o6mi4icxAvCGosW8ygIjPDYIt0bv+HDHccaUpMcOlpMKqcq7Xs1U6uIjyjuUCAjqkr8vYzySmRsAbzxcGYfpo1uL3hNLX+WOxL2Mw2gRC98sLYD8pTD2yilGDveABrAgMBAAECgYAvHh7S5AcFG7jSkXQ2/QwEJJ1jzlmI4w8cVWTyT7CCDGeHqQkKYw+jKNmGY1n8HtwP8CyW5vHKEkt/Irk4ogLb39DHPNbBVrhCn6HbPR/1RdcU3kOWsFnZ8tjx5lxJhP3bSA0J1V2tk/QlZ74RoKMDK59zp2dyjqoqaVIlj7dbaQJBANLPDOGrtIs6n6FcxmGggg5Hy+ajYIxJVcv2yoUfg59Rg3Q/X4QbrxU0NQ4+q/dSZfFSCDJ9lpvJWuO4hZI1MG8CQQDSQYjdlBOePY0pgDE+0jfPh0J5c6Ctyvq7nrps7JsrJpF/FUurUZFCBq63UMIlxFKZuGisngthUBKkkanuiXXFAkEAlBMfxzxm2w1Yp22VY2ntML65uM21uZShkV1Or+eM+tG09mi1XQRdHRXI1rKq9FSKitHGNEEHE3KCR0aLfEGkPQJAK+xq5jc3/ffH1KKmJwSNijoshejE6WmdBbL8KSMoq1QMUyskuq54lP2GZF9nn7Iqu1hN7bm6kOLEckLH6EApLQJBAJkuPP4kB1moPecwPjoW62oNWvdTK59tGHdvVDclnnpbWNurSp63zNGyowmzZn4MaEUduliU9ZuA+g3BuRXnTQo=
生成密钥消耗时间:0.129秒
原文:4028138151b3cf300151b419df0900072015-12-17 11:30:22A01,A02
密文:DnNGSArBjHQ46uyP9aBBs/f+zGwbpM2suh39Imdqu1ebUyCdB6b1AYB8Y7snklttm++EQjrKO8B+06er0+bOJb8lXbHQrvDkFCbcp31TiCBeITZT8bWpHQfwVmutVYKCYGwuk2E5VXQfSySuh6ZsUX/M/3UfS9MvKbQB3d1dyJ8=
加密消耗时间:0.217秒
解密:4028138151b3cf300151b419df0900072015-12-17 11:30:22A01,A02
解密消耗时间:0.003秒
Process finished with exit code 0
来源:https://cloud.tencent.com/developer/article/1640048
0
投稿
猜你喜欢
- 文件上传大部分通过web前端判断后尾名或者service后端判断后尾名,这种操作具有一定的风险,比如:我可以将一个jsp页面,修改后尾名改成
- 本文实例讲述了Jexcel实现按一定规则分割excel文件的方法。分享给大家供大家参考。具体如下:现有一个excel文档,需要读取它并按照一
- 本文实例为大家分享了Android绝对布局AbsoluteLayout的具体代码,供大家参考,具体内容如下1>AbsoluteLayo
- JUC包(java.util.concurrent)中提供了对定时任务的支持,即ScheduledExecutorService接口。本文对
- 上篇文章我们讲解了使用Hibernate Validation来校验数据,当校验完数据后,如果发生错误我们需要给客户返回一个错误信息,因此这
- Android Studio 在引用外部依赖时,发现一直无法引用外部依赖。刚开始以为是墙的问题,尝试修改Gradle配置,未解决问题。最终发
- 上一篇我们详解了setttings.xml的配置项,里面的配置项基本都和仓库有关系,我们使用maven更多的也是要从仓库下载jar包,然后也
- 文章来源:csdn 作者:chensheng913对于Java语言,最体贴的一项设计就是它并没有打算让人们为了写程序而写程序——人们也需要考
- 按官方修改的示例:#MidServerClient.javaimport feign.Param;import org.springfram
- 本文实例讲述了Spring实战之属性覆盖占位符配置器用法。分享给大家供大家参考,具体如下:一 配置文件<?xml version=&q
- 我就废话不多说啦,还是直接看代码吧!/** * 设置数据库是否自动提交事务 * @param flag
- 上一章节回顾:Netty分布式源码分析监听读事件概述pipeline, 顾名思义, 就是管道的意思, 在net
- 概述Spring Boot 监控核心是 spring-boot-starter-actuator 依赖,增加依赖后, Spring Boot
- 一、业务需求实现省份与城市的二级联动二、实现效果三、代码实现1. province_city.jsp前端界面实现<%@ p
- 本文实例讲述了java实现的日期时间转换工具类。分享给大家供大家参考,具体如下:最基础的东西,总结一下,下次用的时候就方便一些了。废话不多说
- 本文帮助大家掌握Java多线程基础知识来对应日后碰到的问题,具体内容如下一、Java多线程面试问题1. 进程和线程之间有什么不同?一个进程是
- 先建个钉钉群,并加好机器人此时,机器人已经添加完毕,接下来编写我们连接机器人小哥的代码import com.alibaba.fastjson
- 委托的Invoke方法用来进行同步调用。同步调用也可以叫阻塞调用,它将阻塞当前线程,然后执行调用,调用完毕后再继续向下进行。同步调用的例子:
- 优点1.观察者和被观察者是抽象耦合的。2.建立一套触发机制。缺点1.如果一个被观察者对象有很多的直接和间接的观察者的话,将所有的观察者都通知
- 实现比较两个List之间的差异,包括获取两List的差集,交集,并集(不去重&去重)的API解法和优化解法的解决方案。求差集/**