RSA公钥验签
1.业务场景,公司做理财业务,但是可能有第三方合作。与第三方合作获得更多客户流量。别人可以在第三方进行购买理财产品。那么怎么保证交易信息的安全性那,我们这里给出rsa加密实现原理。
2.工具类rsa:
公钥私钥的生成百度一下有在线生成的网站。
- 
import java.security.KeyFactory;
 
- 
import java.security.PrivateKey;
 
- 
import java.security.PublicKey;
 
- 
import java.security.Signature;
 
- 
import java.security.spec.PKCS8EncodedKeySpec;
 
- 
import java.security.spec.X509EncodedKeySpec;
 
- 
 
 
- 
import org.apache.commons.codec.binary.Base64;
 
- 
import org.slf4j.Logger;
 
- 
import org.slf4j.LoggerFactory;
 
- 
 
 
- 
import lombok.extern.slf4j.Slf4j;
 
- 
 
 
- 
/**
 
- 
 * 使用私钥将明文进行签名生成秘闻串与明文一起传输。对方收到数据后通过公钥对明文与明文进行验签。
 
- 
 * 若验签通过就说明第一数据没有被修改过;第二这些数据一定是持有私钥的人发送的,因为私钥自己持有,
 
- 
 * 这就起到了防止抵赖的作用。
 
- 
 */
 
- 
@Slf4j
 
- 
public class RSAUtil {
 
- 
    static Logger LOG = LoggerFactory.getLogger(RSAUtil.class);
 
- 
 
 
- 
    private static final String SIGNATURE_ALGORITHM = "SHA1withRSA";                     //签名算法
 
- 
    private static final String KEY_ALGORITHM = "RSA";        //加密算法RSA
 
- 
 
 
- 
    /**
 
- 
     * 公钥验签
 
- 
     *
 
- 
     * @param text      原字符串
 
- 
     * @param sign      签名结果
 
- 
     * @param publicKey 公钥
 
- 
     * @return 验签结果
 
- 
     */
 
- 
    public static boolean verify(String text, String sign, String publicKey) {
 
- 
        try {
 
- 
            Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM);
 
- 
            PublicKey key = KeyFactory.getInstance(KEY_ALGORITHM).generatePublic(new X509EncodedKeySpec(Base64.decodeBase64(publicKey)));
 
- 
            signature.initVerify(key);
 
- 
            signature.update(text.getBytes());
 
- 
            return signature.verify(Base64.decodeBase64(sign));
 
- 
        } catch (Exception e) {
 
- 
            LOG.error("验签失败:text={},sign={}", text, sign, e);
 
- 
        }
 
- 
        return false;
 
- 
    }
 
- 
 
 
- 
    /**
 
- 
     * 签名字符串
 
- 
     *
 
- 
     * @param text       需要签名的字符串
 
- 
     * @param privateKey 私钥(BASE64编码)
 
- 
     * @return 签名结果(BASE64编码)
 
- 
     */
 
- 
    public static String sign(String text, String privateKey) {
 
- 
        byte[] keyBytes = Base64.decodeBase64(privateKey);
 
- 
        PKCS8EncodedKeySpec pkcs8KeySpec = new PKCS8EncodedKeySpec(keyBytes);
 
- 
        try {
 
- 
            KeyFactory keyFactory = KeyFactory.getInstance(KEY_ALGORITHM);
 
- 
            PrivateKey privateK = keyFactory.generatePrivate(pkcs8KeySpec);
 
- 
            Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM);
 
- 
            signature.initSign(privateK);
 
- 
            signature.update(text.getBytes());
 
- 
            byte[] result = signature.sign();
 
- 
            return Base64.encodeBase64String(result);
 
- 
        } catch (Exception e) {
 
- 
            LOG.error("签名失败,text={}", text, e);
 
- 
        }
 
- 
        return null;
 
- 
    }
 
- 
    
 
- 
    private static final String publicKey = "aaaaaaaaa" ;
 
- 
    private static final String privateKey = "bbbbbbbbb";
 
- 
    
 
- 
    public static void main(String[] args) {
 
- 
		String text = "hello world";
 
- 
		String sign = RSAUtil.sign(text, privateKey);
 
- 
		log.info(sign);
 
- 
		boolean verify = RSAUtil.verify(text, sign, publicKey);
 
- 
		log.info("result: {}",verify);
 
- 
    }
 
- 
}
 
3.对数据进行加密之前,首先要保证数据的顺序一致性,顺序不一致可能会导致生成的密文不同。我们这里默认采用拼音排序。
- 
import com.fasterxml.jackson.annotation.JsonInclude;
 
- 
import com.fasterxml.jackson.annotation.JsonPropertyOrder;
 
- 
import com.happylaishop.admin.util.JsonUtil;
 
- 
 
 
- 
/**
 
- 
 * 签名明文,字典即拼音进行排序
 
- 
 */
 
- 
@JsonInclude(JsonInclude.Include.NON_NULL)
 
- 
@JsonPropertyOrder(alphabetic = true)
 
- 
public interface SignText {
 
- 
    default String toText(){
 
- 
        return JsonUtil.obj2String(this);
 
- 
    }
 
- 
}
 
4.采用AOP进行校验。请求方法中前三个参数是string, string SignText的我们进行拦截校验。
authId表示认证对象id,双方约定好。
sign 表示密文签名
text 传输的数据
- 
import org.aspectj.lang.annotation.Aspect;
 
- 
import org.aspectj.lang.annotation.Before;
 
- 
import org.springframework.beans.factory.annotation.Autowired;
 
- 
import org.springframework.stereotype.Component;
 
- 
import org.springframework.util.Assert;
 
- 
 
 
- 
import xxx.util.RSAUtil;
 
- 
 
 
- 
/**
 
- 
 * 验签aop
 
- 
 */
 
- 
@Component
 
- 
@Aspect
 
- 
public class SignAop {
 
- 
 
 
- 
    @Autowired
 
- 
    private SignService signService;
 
- 
 
 
- 
    @Before(value = "execution(* com.happylaishop.admin.controller.*.*(..)) && args(authId,sign,text,..)")
 
- 
    public void verify(String authId,String sign,SignText text){
 
- 
    	/**根据认证id获取对应的公钥*/
 
- 
        String publicKey = signService.getPublicKey(authId);
 
- 
        //拿到公钥之后验签,若验签通过,执行后续业务逻辑,否则报异常
 
- 
        Assert.isTrue(RSAUtil.verify(text.toText(),sign,publicKey),"验签失败");
 
- 
    }
 
- 
}
 
- 
import org.springframework.stereotype.Service;
 
- 
 
 
- 
import java.util.HashMap;
 
- 
import java.util.Map;
 
- 
 
 
- 
/**
 
- 
 * 签名服务
 
- 
 */
 
- 
@Service
 
- 
public class SignService {
 
- 
    static Map<String,String> PUBLIC_KEYS = new HashMap<>();
 
- 
    static {
 
- 
        PUBLIC_KEYS.put("1000","aaaaaaaa");
 
- 
    }
 
- 
 
 
- 
    /**
 
- 
     * 根据授权编号获取公钥
 
- 
     * @param authId
 
- 
     * @return
 
- 
     */
 
- 
    public String getPublicKey(String authId){
 
- 
        return PUBLIC_KEYS.get(authId);
 
- 
    }
 
- 
}
 
我们这里给出一个传输数据,订单对象;
- 
import java.math.BigDecimal;
 
- 
import java.util.Date;
 
- 
 
 
- 
import com.fasterxml.jackson.annotation.JsonFormat;
 
- 
 
 
- 
import lombok.Getter;
 
- 
import lombok.Setter;
 
- 
import lombok.ToString;
 
- 
import lombok.extern.slf4j.Slf4j;
 
- 
 
 
- 
@Getter
 
- 
@Setter
 
- 
@ToString
 
- 
public class OrderParam  implements SignText{
 
- 
	
 
- 
    private String chanId;
 
- 
 
 
- 
    private String chanUserId;
 
- 
 
 
- 
    private String productId;
 
- 
 
 
- 
    private BigDecimal amount;
 
- 
 
 
- 
    private String outerOrderId;
 
- 
 
 
- 
    private String memo;
 
- 
 
 
- 
    @JsonFormat(pattern = "YYYY-MM-DD HH:mm:ss")
 
- 
    private Date createAt;
 
- 
    
 
- 
    public static void main(String[] args) {
 
- 
		OrderParam order = new OrderParam();
 
- 
		order.setAmount(new BigDecimal("100"));
 
- 
		order.setChanId("1000");
 
- 
		order.setChanUserId("1000");
 
- 
		order.setProductId("1");
 
- 
		SignText signText = order;
 
- 
		String text = signText.toText();
 
- 
		System.out.println(text);
 
- 
   }
 
- 
}
 
RSA公钥验签的更多相关文章
- java RSA实现私钥签名、公钥验签、私钥加密数据、公钥解密数据
		
通过OpenSSL生成公私钥文件(如果没有OpenSSL工具建议下载Cmder工具自带OpenSSL指令) 1.生成RSA密钥的方法 genrsa -out private-rsa.key 2048 ...
 - 【绝迹篇】RSA加密算法(私钥加签公钥验签)
		
对于上上篇博客中我讲的一个故事,本文引用: https://www.cnblogs.com/ButterflyEffect/p/9851403.html 故事中提到的关于加密会出现,私钥加密,公钥解密 ...
 - RSA加解密&RSA加验签详解
		
RSA 加密算法是目前最有影响力的 公钥加密算法,并且被普遍认为是目前 最优秀的公钥方案 之一.RSA 是第一个能同时用于 加密 和 数字签名 的算法,它能够 抵抗 到目前为止已知的 所有密码攻击,已 ...
 - RSA签名验签
		
import android.util.Base64; import java.security.KeyFactory; import java.security.PrivateKey; import ...
 - SHA256withRSA证书签名,私钥签名/公钥验签
		
证书签名 package test; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundE ...
 - RSA签名验签学习笔记
		
RSA私钥签名时要基于某个HASH算法,比如MD5或者SHA1等.之前我一直认为签名的过程是:先对明文做HASH计算,然后用私钥直接对HASH值加密.最近才发现不是那么简单,需要对HASH后的数据进行 ...
 - .net core AES加密解密及RSA 签名验签
		
引用 using Org.BouncyCastle.Crypto.Parameters; using Org.BouncyCastle.Security; using System; using Sy ...
 - JAVA/PHP/C#版RSA验签--转
		
本文是上一篇文章的兄弟篇,上篇文章介绍了客户端的sdk中如何基于JAVA/PHP/C#使用RSA私钥签名,然后服务端基于JAVA使用RSA公钥验签,客户端签名/服务端验签的模式只能帮助服务端检查客户端 ...
 - RSA公钥、私钥、签名和验签
		
1 RSA加密算法介绍 RSA又叫非对称加密算法,这类加密算法有一对秘钥,其中一个用来加密一个用来解密.这一对秘钥中你可以选择一个作为私钥(自己保存),另一个作为公钥(对外公开).用私钥加密的内容只能 ...
 
随机推荐
- js02 变量数据类型
			
变量 JavaScript 是一种弱类型的脚本语言 var c = 3:即变量的声明(变量使用之前必须加var声明,编程规范) 变量的命名规则! 1.变量命名必须以字母或是下标符号”_”或者”$”为开 ...
 - RelativeLayout-属性大全
			
// 相对于给定ID控件 <!--将该控件的底部置于给定ID的控件之上--> android:layout_above <!--将该控件的底部置于给定ID的控件之下--> an ...
 - try {}里有一个return语句 finally执行顺序
			
先看例子 package example; class Demo{ public static void main(String args[]) { int x=1; System.out.print ...
 - 编写一个程序,把指定目录下的所有的带.java文件都拷贝到另一个目录中,拷贝成功后,把后缀名是.java的改成.txt。
			
package example; import java.io.*; public class Test { public static void main(String[] args) throws ...
 - Linux架设Jsp环境
			
本文已发表在2010年<网管员世界> 650) this.width=650;" onclick="window.open("http://blog.51ct ...
 - Java学习笔记四
			
1.简介.进程和线程:简单的说就是进程负责为程序开辟内存空间,线程负责具体的执行单元(执行路径). 一个进程中可以有多个执行路径,称为多线程.CPU一次只能执行一个进程,但是一个进程内部可以有多个线程 ...
 - Java学习笔记七 常用API对象三
			
一.泛型:简单说就是对对象类型进行限定的技术 public class GenericDemo { public static void main(String[] args){ /*泛型作为1.5版 ...
 - javascript 幻灯片代码(含自动播放)
			
HTML <div class="slideshow-container"> <div class="mySlides fade"> & ...
 - 使用网络TCP搭建一个简单文件下载器
			
说明:该篇博客是博主一字一码编写的,实属不易,请尊重原创,谢谢大家! 目录 一丶项目介绍 二丶服务器Server 三丶测试TCP server服务器 四丶客户端Client 五丶测试客户端向服务器下载 ...
 - 小米开源文件管理器MiCodeFileExplorer-源码研究(0)-初步研究
			
2011年对着书本Android应用开发揭秘,写了2个月的HelloWorld. 现在想复习并深入,我没有耐心再去一点点地敲代码了. 4年前自己是个学生,实习,现在有工作,只能业余时间研究. ...