Java RSA加密解密及签名验证

一、简介

RSA公开密钥密码体制是一种使用不同的加密密钥与解密密钥,在公开密钥密码体制中,加密密钥(即公开密钥)PK是公开信息,而解密密钥(即秘密密钥)SK是需要保密的。加密算法E和解密算法D也都是公开的。虽然解密密钥SK是由公开密钥PK决定的,但却不能根据PK计算出SK。
正是基于这种理论,1978年出现了著名的RSA算法,它通常是先生成一对RSA密钥,其中之一是保密密钥,由用户保存;另一个为公开密钥,可对外公开,甚至可在网络服务器中注册。为提高保密强度,RSA密钥至少为500位长。这就使加密的计算量很大。为减少计算量,在传送信息时,常采用传统加密方法与公开密钥加密方法相结合的方式,即信息采用改进的DES或IDEA对话密钥加密,然后使用RSA密钥加密对话密钥和信息摘要。对方收到信息后,用不同的密钥解密并可核对信息摘要 。
RSA是被研究得最广泛的公钥算法,从提出到现在已近三十年,经历了各种攻击的考验,逐渐为人们接受,普遍认为是目前最优秀的公钥方案之一。

二、应用

1、密匙生成器,自动生成gongshi公钥和私钥,其中1024为密匙大小,密钥可以设置的范围是96-1024

     keyPairGen.initialize(1024,new SecureRandom());

2、公钥加密,先获取公钥publicKey,进行cipher初始化,对data进行加密。

    cipher.init(Cipher.ENCRYPT_MODE, publicKey);
    Base64.getEncoder().encodeToString(cipher.doFinal(data.getBytes("UTF-8")));

3、私钥解密,先获取私钥privateKey,进行cipher初始化,对data进行转换,在进行解密。

    cipher.init(Cipher.ENCRYPT_MODE, privateKey);
    byte[] inputByte = Base64.getDecoder().decode(data.getBytes("UTF-8"));
    new String(cipher.doFinal(inputByte),"UTF-8");

三、代码

1、运行代码:

package demo;

import java.nio.charset.StandardCharsets;
import java.security.Key;
import java.security.KeyPair;
import java.security.KeyPairGenerator;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.security.SecureRandom;
import java.security.Signature;
import java.util.Base64;

import javax.crypto.Cipher;


public class RSAUtil {

	//自动创建生成公钥和私钥对(RSA算法)
	public static KeyPair getKeyPair() throws Exception {
	    KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA");		  
	    keyPairGen.initialize(1024,new SecureRandom());
	    KeyPair keyPair = keyPairGen.generateKeyPair();
	    return keyPair;
	}

	//公钥加密
	public static String encryption(String data, Key publicKey) throws Exception{
        Cipher cipher = Cipher.getInstance("RSA");
        cipher.init(Cipher.ENCRYPT_MODE, publicKey);
        String information = Base64.getEncoder().encodeToString(cipher.doFinal(data.getBytes("UTF-8")));
        return information;
    }
	
	//私钥解密
	 public static String decrypt(String data, Key privateKey) throws Exception{     
        Cipher cipher = Cipher.getInstance("RSA");
        cipher.init(Cipher.DECRYPT_MODE, privateKey);
        byte[] inputByte = Base64.getDecoder().decode(data.getBytes("UTF-8"));
        String information = new String(cipher.doFinal(inputByte),"UTF-8");
        return information;
    }
	 
	//私钥生成签名 
	public static String sign(String data, PrivateKey privateKey) throws Exception {
	    Signature signature = Signature.getInstance("SHA256withRSA");
	    signature.initSign(privateKey);
	    signature.update(data.getBytes(StandardCharsets.UTF_8));
	    byte[] inputByte = signature.sign();
	    return Base64.getEncoder().encodeToString(inputByte);
	}
	
	//公钥验证签名
	public static boolean validate(String data, String sign, PublicKey publicKey) throws Exception {
	     Signature signature = Signature.getInstance("SHA256withRSA");
	     signature.initVerify(publicKey);
	     signature.update(data.getBytes(StandardCharsets.UTF_8));
	     byte[] inputByte = Base64.getDecoder().decode(sign);
	     return signature.verify(inputByte);
	 }
	 
	 

	public static void main(String[] args) throws Exception {
		System.out.println("====================================================================");
		   
	    KeyPair keyPair = getKeyPair();
	    PublicKey publicKey =  keyPair.getPublic();
	    PrivateKey privateKey = keyPair.getPrivate();
	    
	    String data0 = "RSA加密解密成功!";
	    System.out.println("==公钥:"+keyPair.getPublic());
	    System.out.println("==私钥:"+keyPair.getPrivate());
	    
	    System.out.println("加密前数据:"+data0);
	    
	    String data1 = encryption(data0,publicKey);
	    System.out.println("加密后数据:"+data1);
	    
	    String messageDe = decrypt(data1,privateKey);
	    System.out.println("解密后数据:"+messageDe);
	    
	    String sign = sign(data0, keyPair.getPrivate());
		boolean validate = validate(data0,sign, keyPair.getPublic());
		System.out.println("校验通过:"+validate);
	    System.out.println("====================================================================");

	}

}


运行结果:
在这里插入图片描述

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
RSA加密签名都是基于RSA算法的,但是加密签名的目的和流程有所不同。 RSA加密流程如下: 1. 选择两个大质数p和q,计算n=p*q,并记为RSA模数。 2. 计算欧拉函数φ(n)=(p-1)*(q-1)。 3. 选择一个整数e,1<e<φ(n),且e与φ(n)互质,e作为公钥指数。 4. 计算d,使得d*e≡1(mod φ(n)),d作为私钥指数。 5. 加密时,对明文m进行加密,计算c=m^e(mod n)。 6. 解密时,对密文c进行解密,计算m=c^d(mod n)。 RSA签名流程如下: 1. 选择两个大质数p和q,计算n=p*q,并记为RSA模数。 2. 计算欧拉函数φ(n)=(p-1)*(q-1)。 3. 选择一个整数e,1<e<φ(n),且e与φ(n)互质,e作为签名者的私钥指数。 4. 计算d,使得d*e≡1(mod φ(n)),d作为签名者的公钥指数。 5. 对消息m进行签名,计算s=m^d(mod n)。 6. 验证签名时,对接收到的消息m和签名s进行验证,计算m'=s^e(mod n),如果m'=m,则验证通过。 下面是Java代码示例: RSA加密: ```java import java.security.KeyPair; import java.security.KeyPairGenerator; import java.security.PrivateKey; import java.security.PublicKey; import javax.crypto.Cipher; public class RSAEncrypt { public static void main(String[] args) throws Exception { String plainText = "Hello, world!"; KeyPairGenerator kpg = KeyPairGenerator.getInstance("RSA"); kpg.initialize(2048); KeyPair kp = kpg.generateKeyPair(); PublicKey publicKey = kp.getPublic(); PrivateKey privateKey = kp.getPrivate(); Cipher cipher = Cipher.getInstance("RSA"); cipher.init(Cipher.ENCRYPT_MODE, publicKey); byte[] encryptedText = cipher.doFinal(plainText.getBytes()); System.out.println("加密后的密文:" + new String(encryptedText)); cipher.init(Cipher.DECRYPT_MODE, privateKey); byte[] decryptedText = cipher.doFinal(encryptedText); System.out.println("解密后的明文:" + new String(decryptedText)); } } ``` RSA签名: ```java import java.security.KeyPair; import java.security.KeyPairGenerator; import java.security.PrivateKey; import java.security.PublicKey; import java.security.Signature; public class RSASign { public static void main(String[] args) throws Exception { String plainText = "Hello, world!"; KeyPairGenerator kpg = KeyPairGenerator.getInstance("RSA"); kpg.initialize(2048); KeyPair kp = kpg.generateKeyPair(); PublicKey publicKey = kp.getPublic(); PrivateKey privateKey = kp.getPrivate(); Signature signature = Signature.getInstance("SHA256withRSA"); signature.initSign(privateKey); signature.update(plainText.getBytes()); byte[] signedText = signature.sign(); System.out.println("签名后的内容:" + new String(signedText)); signature.initVerify(publicKey); signature.update(plainText.getBytes()); boolean result = signature.verify(signedText); System.out.println(result ? "验签成功" : "验签失败"); } } ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

傻猴儿

小编,多谢客官留下的赏钱。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值