基於hutool實現非對稱加密(RSA和國密SM2)的加解密和加簽驗簽

来源:https://www.cnblogs.com/hdwang/archive/2022/05/25/16310314.html
-Advertisement-
Play Games

背景 對外服務的介面為了安全起見,往往需要進行相應的安全處理:數據加密傳輸和身份認證。數據加密傳輸有對稱加密和非對稱加密兩種,為了更加安全起見採用非對稱加密比較好些,身份認證則採用數字簽名可以實現。 非對稱加密缺點:加解密速度慢、RSA有最大長度要求。 方案一 僅採用非對稱加密 RSA對內容長度的要 ...


背景

對外服務的介面為了安全起見,往往需要進行相應的安全處理:數據加密傳輸和身份認證。數據加密傳輸有對稱加密和非對稱加密兩種,為了更加安全起見採用非對稱加密比較好些,身份認證則採用數字簽名可以實現。

非對稱加密缺點:加解密速度慢、RSA有最大長度要求。

 

方案一

僅採用非對稱加密

RSA對內容長度的要求可以通過分組加解密解決

參考:https://blog.csdn.net/draven1122/article/details/55212252

 

方案二

非對稱加密+對稱加密

內容採用對稱加密(AES)加密,非對稱加密僅加密AES密鑰

調用方

  1.簽名生成:介面參數->摘要演算法(SHA)->參數摘要->簽名(自己RSA私鑰)->簽名

  2.內容(介面參數 + 簽名 ) -->對稱加密(AES)--> 內容密文

  3.AES密鑰-->非對稱加密(對方RSA公鑰)--> AES密鑰密文

  4. 內容密文(請求體)+AES密鑰密文(請求頭)-> 傳輸給接收方

接收方

 1. 獲取AES密鑰密文-> 非對稱解密(自己RSA私鑰)->AES密鑰

 2. 內容密文-> AES密鑰-> 內容明文

 3. 驗證簽名:內容里的參數->摘要演算法(SHA)->參數摘要->驗簽(對方RSA公鑰)->簽名

 4. 驗簽通過,繼續執行介面

 5. 返回值(加解密可選)

 

加密工具類可以使用:https://hutool.cn/docs/#/crypto/%E6%A6%82%E8%BF%B0 

基於hutool中密碼工具類實現的rsa和國密的非對稱加解密演算法和加簽驗簽演算法代碼如下

maven依賴

 <!-- huTool工具箱 -->
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>5.7.22</version>
        </dependency>

        <dependency>
            <groupId>org.bouncycastle</groupId>
            <artifactId>bcprov-jdk15to18</artifactId>
            <version>1.69</version>
        </dependency>

rsa非對稱加密

package com.hdwang.test.hutool;

import cn.hutool.core.util.CharsetUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.crypto.SecureUtil;
import cn.hutool.crypto.asymmetric.KeyType;
import cn.hutool.crypto.asymmetric.RSA;
import cn.hutool.crypto.asymmetric.Sign;
import cn.hutool.crypto.asymmetric.SignAlgorithm;

import java.nio.charset.StandardCharsets;
import java.security.KeyPair;
import java.security.PrivateKey;
import java.security.PublicKey;
import java.util.Base64;

/**
 * @author wanghuidong
 * @date 2022/5/25 21:00
 */
public class RsaTest {

    public static void main(String[] args) {
        String text = "人最寶貴的是生命.生命對每個人只有一次.人的一生應當這樣度過:當他迴首往事的時候,不會因為虛度年華而悔恨,也不會因為碌碌無為而羞恥.這樣,在臨死的時候,他能夠說:“我已把自己的整個的生命和全部的精力獻給了世界上最壯麗的事業---------為人類的解放而鬥爭.”";
        System.out.println("原文:" + text);

        //生成公私鑰對
        KeyPair pair = SecureUtil.generateKeyPair("RSA");
        PrivateKey privateKey = pair.getPrivate();
        PublicKey publicKey = pair.getPublic();
        //獲得私鑰
        String privateKeyStr = bytesToBase64(privateKey.getEncoded());
        System.out.println("私鑰:" + privateKeyStr);
        //獲得公鑰
        String publicKeyStr = bytesToBase64(publicKey.getEncoded());
        System.out.println("公鑰:" + publicKeyStr);

        RSA rsa = new RSA(privateKeyStr, publicKeyStr);
        System.out.println(rsa);

        //公鑰加密,私鑰解密
        byte[] encrypt = rsa.encrypt(StrUtil.bytes(text, CharsetUtil.CHARSET_UTF_8), KeyType.PublicKey);
        System.out.println("公鑰加密:" + bytesToBase64(encrypt));

        byte[] decrypt = rsa.decrypt(encrypt, KeyType.PrivateKey);
        System.out.println("私鑰解密:" + new String(decrypt,StandardCharsets.UTF_8));

//        //私鑰加密,公鑰解密
//        byte[] encrypt2 = rsa.encrypt(StrUtil.bytes(text, CharsetUtil.CHARSET_UTF_8), KeyType.PrivateKey);
//        System.out.println("私鑰加密:" + bytesToBase64(encrypt2));
//        byte[] decrypt2 = rsa.decrypt(encrypt2, KeyType.PublicKey);
//        System.out.println("公鑰解密:" + bytesToBase64(decrypt2));

        Sign sign = SecureUtil.sign(SignAlgorithm.SHA256withRSA, privateKeyStr, publicKeyStr);
        //簽名
        byte[] data = text.getBytes(StandardCharsets.UTF_8);
        byte[] signed = sign.sign(data);
        String signedStr = bytesToBase64(signed);
        System.out.println("簽名:" + signedStr);

        //驗證簽名
        boolean verify = sign.verify(data, base64ToBytes(signedStr));
        System.out.println("驗簽:" + verify);

    }

    /**
     * 位元組數組轉Base64編碼
     *
     * @param bytes 位元組數組
     * @return Base64編碼
     */
    private static String bytesToBase64(byte[] bytes) {
        byte[] encodedBytes = Base64.getEncoder().encode(bytes);
        return new String(encodedBytes, StandardCharsets.UTF_8);
    }

    /**
     * Base64編碼轉位元組數組
     *
     * @param base64Str Base64編碼
     * @return 位元組數組
     */
    private static byte[] base64ToBytes(String base64Str) {
        byte[] bytes = base64Str.getBytes(StandardCharsets.UTF_8);
        return Base64.getDecoder().decode(bytes);
    }
}

國密非對稱加密(SM2)

package com.hdwang.test.hutool;

import cn.hutool.core.util.HexUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.crypto.SecureUtil;
import cn.hutool.crypto.SmUtil;
import cn.hutool.crypto.asymmetric.KeyType;
import cn.hutool.crypto.asymmetric.SM2;

import java.nio.charset.StandardCharsets;
import java.security.KeyPair;
import java.util.Base64;

/**
 * 國密非對稱加解密和加簽驗簽演算法
 *
 * @author wanghuidong
 * @date 2022/5/25 20:50
 */
public class SmTest {
    public static void main(String[] args) {
        String text = "人最寶貴的是生命.生命對每個人只有一次.人的一生應當這樣度過:當他迴首往事的時候,不會因為虛度年華而悔恨,也不會因為碌碌無為而羞恥.這樣,在臨死的時候,他能夠說:“我已把自己的整個的生命和全部的精力獻給了世界上最壯麗的事業---------為人類的解放而鬥爭.”";
        System.out.println("原文:" + text);

        KeyPair pair = SecureUtil.generateKeyPair("SM2");
        byte[] privateKey = pair.getPrivate().getEncoded();
        byte[] publicKey = pair.getPublic().getEncoded();
        System.out.println("公鑰:\n" + bytesToBase64(publicKey));
        System.out.println("私鑰:\n" + bytesToBase64(privateKey));

        SM2 sm2 = SmUtil.sm2(privateKey, publicKey);
        // 公鑰加密,私鑰解密
        String encryptStr = sm2.encryptBcd(text, KeyType.PublicKey);
        System.out.println("加密後:" + encryptStr);

        String decryptStr = StrUtil.utf8Str(sm2.decryptFromBcd(encryptStr, KeyType.PrivateKey));
        System.out.println("解密後:" + decryptStr);
        //加簽
        String sign = sm2.signHex(HexUtil.encodeHexStr(text));
        System.out.println("簽名:" + sign);
        //驗簽
        boolean verify = sm2.verifyHex(HexUtil.encodeHexStr(text), sign);
        System.out.println("驗簽:" + verify);

    }


    /**
     * 位元組數組轉Base64編碼
     *
     * @param bytes 位元組數組
     * @return Base64編碼
     */
    private static String bytesToBase64(byte[] bytes) {
        byte[] encodedBytes = Base64.getEncoder().encode(bytes);
        return new String(encodedBytes, StandardCharsets.UTF_8);
    }

    /**
     * Base64編碼轉位元組數組
     *
     * @param base64Str Base64編碼
     * @return 位元組數組
     */
    private static byte[] base64ToBytes(String base64Str) {
        byte[] bytes = base64Str.getBytes(StandardCharsets.UTF_8);
        return Base64.getDecoder().decode(bytes);
    }
}

 

 

參考文章:

https://www.cnblogs.com/pcheng/p/9629621.html

https://blog.csdn.net/woniu211111/article/details/108114402

https://blog.csdn.net/draven1122/article/details/55212252


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 本文記錄如何在 Vue2 環境下儘量使用 Vue3 的 Composition-api 並配合 Vuetify2 使用 ...
  • 切麵:公共的,通用的,重覆的功能稱為切麵,面向切麵編程就是將切麵提取出來,單獨開發,在需要調用的方法中通過動態代理的方式進行織入 ...
  • 1.微服務架構是什麼? 由很多個微服務來組成,並且每個服務與服務有自己獨立的資料庫,服務與服務之間是通過輕量級的協議(比如:rpc,http等)來調用,每個服務是獨立的,自治的能夠獨立的發佈和部署。如下圖: 本質上來講,微服務只是一種架構風格。架構風格如何採取,那麼就需要涉及到架構特征這個概念。而架 ...
  • 1. Java基礎(1)——ThreadLocal 1.1. ThreadLocal ThreadLocal是一個泛型類,當我們在一個類中聲明一個欄位:private ThreadLocal<Foo> threadLocalFoo = new ThreadLocal<>();時,這時候,即使不同的線 ...
  • 事務原本是資料庫中的概念,在實際項目的開發中,進行事務的處理一般是在業務邏輯層, 即 Service 層。這樣做是為了能夠使用事務的特性來管理關聯操作的業務。 ...
  • 第一章 : 開始 1.1 編寫一個簡單的C++程式 要點:每個函數有且只能擁有一個main函數,且main的預設返回是一個int類型。 函數定義:返回類型,函數名,形參列表,函數體 ###1.1.1編譯 運行程式 ####程式源文件命名約定 無論使用命令行還是IDE(集成開發環境),都將代碼存放在一 ...
  • 對JavaSE_關鍵字 介面 代碼塊 枚舉 相關重要知識點進行了介紹和總結 ...
  • Spring框架是一個容器,是一個整合了很多其他框架的框架,它的核心是 IOC 和 AOP ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...