轻装上阵,不调用jar包,用C#写SM4加密算法【卸载IKVM 】

news2025/2/12 0:54:12

前言

记得之前写了一个文章,是关于java和c#加密不一致导致需要使用ikvm的方式来进行数据加密,主要是ikvm把打包后的jar包打成dll包,然后Nuget引入ikvm,从而实现算法的统一,这几天闲来无事,网上找了一下加密库【BouncyCastle.dll】进行加密,目的是想统一加密。因为ikvm相对重了点,引入一堆dll包。

官方网址

c#入口: https://www.bouncycastle.org/csharp/

java入口: https://www.bouncycastle.org/java.html

在这里插入图片描述
(如图所示)在1.8.4中 发现它是支持SM4 加密的,如果你想要使用SM4加密算法,最低版本需要是1.8.4。
在这里插入图片描述

Nuget安装(如图所示)

注意事项

这个是针对java和c#的加解密一致性

1.算法要保持一致 均是SM4
2.算法模式要保持一致 (如CBC和ECB 当然还有填充模式)
举个例子:我这边就使用 SM4/CBC/PKCS5Padding
3.编码一致性
如果JAVA 加解密用的UTF8 ,C#加解密用的是GBK 这样肯定不行了
俺们都是Chinese,所以果断选择UTF8了

最后就是代码部分了

java

需要引入bcprov-jdk15on-1.59.jar 和 httpcore-4.4.3.jar

package com.ken.utils;
import java.security.*;
import java.util.Arrays;
import javax.crypto.Cipher;
import javax.crypto.KeyGenerator;
import javax.crypto.spec.SecretKeySpec;
import org.bouncycastle.jce.provider.BouncyCastleProvider;
import org.bouncycastle.pqc.math.linearalgebra.ByteUtils;


public class SM4Util {

    static {
        Security.addProvider(new BouncyCastleProvider());
    }
    
	//默认是UTF-8编码
    //private static final String ENCODING = "UTF-8";
    public static final String ALGORITHM_NAME = "SM4";
    // 加密算法/分组加密模式/分组填充方式
    // PKCS5Padding-以8个字节为一组进行分组加密
    // 定义分组加密模式使用:PKCS5Padding
    public static final String ALGORITHM_NAME_ECB_PADDING = "SM4/ECB/PKCS5Padding";
    // 128 32位16进制  本身SM4默认就是这一种
   // public static final int DEFAULT_KEY_SIZE = 128;

  // 这边我默认了密钥
    public static final String SM4_KEY = "86C63180C2806ED1F47B859DE501215B";

    /**
     * 生成ECB暗号
     * @explain ECB模式(电子密码本模式:Electronic codebook)
     * @param algorithmName
     *            算法名称
     * @param mode
     *            模式
     * @param key
     * @return
     * @throws Exception
     */
    private static Cipher generateEcbCipher(String algorithmName, int mode, byte[] key) throws Exception {
        Cipher cipher = Cipher.getInstance(algorithmName, BouncyCastleProvider.PROVIDER_NAME);
        Key sm4Key = new SecretKeySpec(key, ALGORITHM_NAME);
        cipher.init(mode, sm4Key);
        return cipher;
    }


    /** 方式一:系统生成密钥
     * 自动生成密钥
     * @explain
     * @return
     * @throws NoSuchAlgorithmException
     * @throws NoSuchProviderException
     */
    public static byte[] generateKey() throws Exception {
        return generateKey(DEFAULT_KEY_SIZE);
    }

    /**
     * @explain
     * @param keySize
     * @return
     * @throws Exception
     */
    public static byte[] generateKey(int keySize) throws Exception {
        KeyGenerator kg = KeyGenerator.getInstance(ALGORITHM_NAME, BouncyCastleProvider.PROVIDER_NAME);
        kg.init(keySize, new SecureRandom());
        return kg.generateKey().getEncoded();
    }


    /**
     * sm4加密
     * @explain 加密模式:ECB
     *          密文长度不固定,会随着被加密字符串长度的变化而变化
     * @param hexKey
     *            16进制密钥(忽略大小写)
     * @param paramStr
     *            待加密字符串
     * @return 返回16进制的加密字符串
     * @throws Exception
     */
    public static String encryptEcb(String hexKey, String paramStr) throws Exception {
        String cipherText = "";
        // 16进制字符串-->byte[]
        byte[] keyData = ByteUtils.fromHexString(hexKey);
        // String-->byte[]
        byte[] srcData = paramStr.getBytes(ENCODING);
        // 加密后的数组
        byte[] cipherArray = encrypt_Ecb_Padding(keyData, srcData);
        // byte[]-->hexString
        cipherText = ByteUtils.toHexString(cipherArray);
        return cipherText;
    }

    /**
     * 加密模式之Ecb 方法二:自己提供16进制的密钥
     * @explain
     * @param key
     * @param data
     * @return
     * @throws Exception
     */
    public static byte[] encrypt_Ecb_Padding(byte[] key, byte[] data) throws Exception {
        Cipher cipher = generateEcbCipher(ALGORITHM_NAME_ECB_PADDING, Cipher.ENCRYPT_MODE, key);
        return cipher.doFinal(data);
    }


    /**
     * sm4解密
     * @explain 解密模式:采用ECB
     * @param hexKey
     *            16进制密钥
     * @param cipherText
     *            16进制的加密字符串(忽略大小写)
     * @return 解密后的字符串
     * @throws Exception
     */
    public static String decryptEcb(String hexKey, String cipherText) throws Exception {
        // 用于接收解密后的字符串
        String decryptStr = "";
        // hexString-->byte[]
        byte[] keyData = ByteUtils.fromHexString(hexKey);
        // hexString-->byte[]
        byte[] cipherData = ByteUtils.fromHexString(cipherText);
        // 解密
        byte[] srcData = decrypt_Ecb_Padding(keyData, cipherData);
        // byte[]-->String
        decryptStr = new String(srcData, ENCODING);
        return decryptStr;
    }

    /**
     * 解密
     * @explain
     * @param key
     * @param cipherText
     * @return
     * @throws Exception
     */
    public static byte[] decrypt_Ecb_Padding(byte[] key, byte[] cipherText) throws Exception {
        Cipher cipher = generateEcbCipher(ALGORITHM_NAME_ECB_PADDING, Cipher.DECRYPT_MODE, key);
        return cipher.doFinal(cipherText);
    }

    /**
     * 校验加密前后的字符串是否为同一数据
     * @explain
     * @param hexKey
     *            16进制密钥(忽略大小写)
     * @param cipherText
     *            16进制加密后的字符串
     * @param paramStr
     *            加密前的字符串
     * @return 是否为同一数据
     * @throws Exception
     */
    public static boolean verifyEcb(String hexKey, String cipherText, String paramStr) throws Exception {
        // 用于接收校验结果
        boolean flag = false;
        // hexString-->byte[]
        byte[] keyData = ByteUtils.fromHexString(hexKey);
        // 将16进制字符串转换成数组
        byte[] cipherData = ByteUtils.fromHexString(cipherText);
        // 解密
        byte[] decryptData = decrypt_Ecb_Padding(keyData, cipherData);
        // 将原字符串转换成byte[]
        byte[] srcData = paramStr.getBytes(ENCODING);
        // 判断2个数组是否一致
        flag = Arrays.equals(decryptData, srcData);
        return flag;
    }


 

    /**
     * 字符串转化成为16进制字符串
     * @param s
     * @return
     */
    public static String strTo16(String s) {
        String str = "";
        for (int i = 0; i < s.length(); i++) {
            int ch = (int) s.charAt(i);
            String s4 = Integer.toHexString(ch);
            str = str + s4;
        }
        return str;
    }



    public static String byteArrayToHexStr(byte[] byteArray) {
        StringBuilder stringBuilder = new StringBuilder("");
        if (byteArray == null || byteArray.length <= 0) {
            return null;
        }
        for (int i = 0; i < byteArray.length; i++) {
            int v = byteArray[i] & 0xFF;
            String hv = Integer.toHexString(v);
            if (hv.length() < 2) {
                stringBuilder.append(0);
            }
            stringBuilder.append(hv);
        }
        return stringBuilder.toString();
    }
	
}

C# (Csharp)

using Org.BouncyCastle.Crypto.IO;
using Org.BouncyCastle.Crypto.Parameters;
using Org.BouncyCastle.Crypto;
using Org.BouncyCastle.Security;
using Org.BouncyCastle.Utilities.Encoders;
using System;
using System.IO;
using System.Text;

namespace Web.Security.Util
{
	public static class SM4Util
	{

		/// <summary>
		/// 默认编码
		/// </summary>
		private static Encoding DefaultEncoding = Encoding.UTF8;
			//"GB2312";

		public const string ALGORITHM_NAME = "SM4";

		/// <summary>
		/// ECB模式 [pkcs5padding填充方式]
		/// </summary>
		public const string ALGORITHM_NAME_ECB_PADDING = "SM4/ECB/PKCS5Padding";

		/// <summary>
		/// CBC模式 [pkcs5padding填充方式]
		/// </summary>
		public const string ALGORITHM_NAME_CBC_PADDING = "SM4/CBC/PKCS5Padding";

		//这个不需要  SM4默认就是128位的密钥
		//public const int DEFAULT_KEY_SIZE = 128;


		/// <summary>
		/// 解密
		/// </summary>
		/// <param name="key">密钥</param>
		/// <param name="passInput"></param>
		/// <param name="encoding">编码</param>
		/// <param name="algorithmMode">默认是ECB模式 [pkcs5padding填充方式]</param>
		/// <returns></returns>
		public static string DecryptEcb(string key, string passInput, Encoding encoding, string algorithmMode= ALGORITHM_NAME_ECB_PADDING)
		{
			encoding = encoding??Encoding.UTF8;
			byte[] keyBytes = Hex.Decode(key);
			byte[] input = Hex.Decode(passInput);
			return encoding.GetString(DecryptEcb(keyBytes, input, algorithmMode));
		}

		/// <summary>
		/// 解密
		/// </summary>
		/// <param name="key">密钥</param>
		/// <param name="passInput"></param>
		/// <param name="algorithmMode">默认是ECB模式 [pkcs5padding填充方式]</param>
		/// <returns></returns>
		public static string DecryptEcb(string key, string passInput, string algorithmMode = ALGORITHM_NAME_ECB_PADDING)
		{
			return DecryptEcb(key, passInput, DefaultEncoding, algorithmMode);
		}

		/// <summary>
		/// 解密
		/// </summary>
		/// <param name="keyBytes">密钥</param>
		/// <param name="passInput"></param>
		/// <param name="algorithmMode">加密方式</param>
		/// <returns></returns>
		/// <exception cref="Exception"></exception>
		public static byte[] DecryptEcb(byte[] keyBytes, byte[] passInput, string algorithmMode)
		{
			KeyParameter key = ParameterUtilities.CreateKeyParameter(ALGORITHM_NAME, keyBytes);
			IBufferedCipher inCipher = CipherUtilities.GetCipher(algorithmMode);
			//forEncryption位false表示解密
			inCipher.Init(false, key);

			MemoryStream bIn = new MemoryStream(passInput, false);
			CipherStream cIn = new CipherStream(bIn, inCipher, null);
			byte[] bytes = new byte[passInput.Length];
			byte[] totalBytes;
			try
			{
				#region 此代码一次性读取解密 可行
				/*BinaryReader dIn = new BinaryReader(cIn);
				byte[] extra = dIn.ReadBytes(passInput.Length);
				Array.Copy(extra, 0, bytes, 0, extra.Length);*/
				#endregion

				#region 官方demo 是先处理一半  再处理剩下的 最后把剩下的复制到bytes剩余部分
				BinaryReader dIn = new BinaryReader(cIn);
				for (int i = 0; i != passInput.Length / 2; i++)
				{
					bytes[i] = dIn.ReadByte();
				}

				int remaining = bytes.Length - passInput.Length / 2;
				byte[] extra = dIn.ReadBytes(remaining);

				//把为了加密补位的部分去掉
				if (extra.Length < remaining)
				{
					int len = passInput.Length/2 + extra.Length;
					totalBytes = new byte[len];
					Array.Copy(bytes, 0, totalBytes, 0, passInput.Length / 2);
					extra.CopyTo(totalBytes, passInput.Length / 2);
					return totalBytes;
				}
				else
				{
					extra.CopyTo(bytes, passInput.Length / 2);
				}
				//throw new EndOfStreamException();
				#endregion
			}
			catch (Exception e)
			{
				throw new Exception("SM4 failed encryption - " + e, e);
			}

			return bytes;
		}

		/// <summary>
		/// 加密
		/// </summary>
		/// <param name="key">密钥</param>
		/// <param name="text"></param>
		/// <param name="encoding">编码</param>
		/// <param name="algorithmMode">默认是ECB模式 [pkcs5padding填充方式]</param>
		/// <returns></returns>
		public static string EncryptEcb(string key, string text, Encoding encoding,string algorithmMode = ALGORITHM_NAME_ECB_PADDING)
		{
			encoding = encoding??Encoding.UTF8;
			byte[] keyBytes = Hex.Decode(key);
			byte[] input = encoding.GetBytes(text);
			return Hex.ToHexString(EncryptEcb(keyBytes, input, algorithmMode));
		}

		/// <summary>
		/// 加密
		/// </summary>
		/// <param name="key">密钥</param>
		/// <param name="text"></param>
		/// <param name="algorithmMode"></param>
		/// <returns></returns>
		public static string EncryptEcb(string key, string text, string algorithmMode = ALGORITHM_NAME_ECB_PADDING)
		{
			return EncryptEcb(key,text, DefaultEncoding, algorithmMode);
		}

		

		/// <summary>
		/// 加密
		/// </summary>
		/// <param name="keyBytes">密钥</param>
		/// <param name="input"></param>
		/// <returns></returns>
		/// <exception cref="Exception"></exception>
		public static byte[] EncryptEcb(byte[] keyBytes, byte[] input,string algorithmMode)
		{
			KeyParameter key = ParameterUtilities.CreateKeyParameter(ALGORITHM_NAME, keyBytes);
			IBufferedCipher outCipher = CipherUtilities.GetCipher(algorithmMode);
			//forEncryption位true表示加密
			outCipher.Init(true, key);
			MemoryStream bOut = new MemoryStream();
			CipherStream cOut = new CipherStream(bOut, null, outCipher);

			try
			{
				//处理前一半
				for (int i = 0; i != input.Length / 2; i++)
				{
					cOut.WriteByte(input[i]);
				}
				//处理后一半
				cOut.Write(input, input.Length / 2, input.Length - input.Length / 2);
				cOut.Close();
			}
			catch (IOException e)
			{
				throw new Exception("SM4 failed encryption - " + e, e);
			}
			byte[] bytes = bOut.ToArray();
			return bytes;
		}		
	}

}

密钥生成

java可以使用UUID生成32位的16进制字符串

C#可以使用 GUID生成32位的16进制字符串

密钥格式(示例): 7A5B5AE03F764358AEAEF0D1B4B2ADAE


调用方式

string SM4_KEY =“7A5B5AE03F764358AEAEF0D1B4B2ADAE”;


java

SM4Util.encryptEcb(SM4_KEY, “待加密文本”);

SM4Util.decryptEcb(SM4_KEY, “加密后的文本”);

C#

SM4Util.EncryptEcb(SM4_KEY, “待加密文本”);
SM4Util.DecryptEcb(SM4_KEY, “加密后的文本”)

ps: java版本的,如果数据量超过5000,不建议全部加密,会比较耗时。c# 2万条依旧扛得住。

这个我试过虽然加密的结果两者(java和C#)是不一样的,但解密的结果都是一样的,如果你对此不放心 也可以把java版的改成使用统一版本的BouncyCastle。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/864382.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于三维模型OSGB格式轻量化压缩出现分层现象分析

关于三维模型OSGB格式轻量化压缩出现分层现象分析 三维模型OSGB格式的轻量化压缩分层现象是指在进行几何压缩和纹理压缩过程中&#xff0c;由于不同部位的信息复杂度存在差异&#xff0c;可能会出现某些部位被处理过度&#xff0c;导致其失去了细节和质感的现象。本文将从几何…

“解锁IDEA的潜力:高级Java Maven项目配置指南”

目录 前言&#xff1a;流程目录&#xff1a;1.确保Java和Maven已安装检查Java是否已正确安装并配置环境变量 2.创建一个新的Maven项目导航到要创建项目的目录配置Maven运行以下命令创建一个新的Maven项目 3.配置项目的pom.xml文件打开项目根目录下的pom.xml文件配置Web.xml 4.配…

什么是绩效管理?绩效管理包括哪些内容?

阅读本文您可以了解&#xff1a;1、绩效管理的定义&#xff1b;2、绩效管理的内容&#xff1b; 一、什么是绩效管理 绩效管理是一种组织和管理方法&#xff0c;旨在确保员工的工作与组织的目标保持一致&#xff0c;以及激励和提高员工的工作表现。它涉及设定明确的目标和标准&…

【Spring】-Spring的IoC和DI

作者&#xff1a;学Java的冬瓜 博客主页&#xff1a;☀冬瓜的主页&#x1f319; 专栏&#xff1a;【Framework】 主要内容&#xff1a;什么是spring&#xff1f;IoC容器是什么&#xff1f;如何使代码解耦合&#xff1f;IoC的核心原理&#xff0c;IoC的优点。依赖注入/对象装配/…

VBA_MF系列技术资料1-152

MF系列VBA技术资料 为了让广大学员在VBA编程中有切实可行的思路及有效的提高自己的编程技巧&#xff0c;我参考大量的资料&#xff0c;并结合自己的经验总结了这份MF系列VBA技术综合资料&#xff0c;而且开放源码&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-04属于定…

沁恒ch32V208处理器开发(三)GPIO控制

目录 GPIO功能概述 CH32V2x 微控制器的GPIO 口可以配置成多种输入或输出模式&#xff0c;内置可关闭的上拉或下拉电阻&#xff0c;可以配置成推挽或开漏功能。GPIO 口还可以复用成其他功能。端口的每个引脚都可以配置成以下的多种模式之一&#xff1a; 1 浮空输入 2 上拉输入…

8年经验之谈 —— 35岁以上的测试开发工程师都去哪里了?

“测试开发工程师就是吃青春饭&#xff01;” “35岁就是测试开发工程师的天花板&#xff0c;没有工作机会了。” “测试开发工程师趁早转行&#xff0c;不然迟早失业。” …… 网上对测试开发工程师不友好的言论非常多。真的是这样吗&#xff1f;如果不是这样&#xff0c;…

Cesium 1.107+ 自定义类支持 readyPromise

由于cesium 1.107 的图元(Primitive) 已经不支持 readyPromise。 但是个人感觉比较好用,于是用了一个插件来实现。 用法: // 定义图元并添加,和之前一样 const boxGreen new BoxPrimitive({color: "#ff0000" }) viewer.scene.primitives.add(boxGreen.primitive)/…

lazada、shopee店铺如何利用测评提高权重和排名?

在 lazada、shopee平台上开店后&#xff0c;卖家们必须对店铺的权重进行更多的关注。如果店铺的权重越高&#xff0c;那么它就会带来更多的流量和更多的订单&#xff0c;那么在 lazada、shopee平台上开设一家店铺&#xff0c;该怎样增加它的店铺权重和排名呢&#xff1f; laza…

算能BM1684X部署手写数字识别模型

大致流程--------------------------------------------------------------- 参考《sophon-sail_zh》 移植步骤------------------------------------------------------------------------ 首先搭建好自己的网络模型&#xff0c;并导出为onnx格式--具体可以参照--> GitH…

aspose 使用ftl模板生成word和pdf

1 先找到word模板&#xff0c;用${}&#xff0c;替换变量&#xff0c;保存&#xff0c;然后另存为xml,最后把xml后缀改成ftl。 如下图&#xff1a; word 模板文件 ftl模板文件如下: 2 代码生成 下面函数将ftl填充数据&#xff0c;并生成word和pdf /*** * param dataMap 模板…

c++继承总结

一 继承的由来 我使用类也有一段时间了&#xff0c;慢慢觉得我们做一件事时&#xff0c;就是要先描述&#xff0c;例如写一个管理系统登记学校成员的信息&#xff0c;我们就要先对在学校内的老师和学生做描述&#xff0c;学生要有年龄&#xff0c;班级&#xff0c;姓名&#xf…

postman接口测试中文汉化教程

想必同学们对于接口测试工具postman的使用并不陌生&#xff0c;以及最近大为流行的国产工具apifox。对于使用过的同学来说&#xff0c;两者区别以及优缺点很容易别展示出来&#xff0c;postman相比apifox来说更加轻量&#xff0c;但是apifox更加符合国人的使用习惯....中国人给…

谷歌云 | BigQuery 现在支持用于查询开放表格式的清单文件

Cloud Ace 是谷歌云全球战略合作伙伴&#xff0c;拥有 300 多名工程师&#xff0c;也是谷歌最高级别合作伙伴&#xff0c;多次获得 Google Cloud 合作伙伴奖。作为谷歌托管服务商&#xff0c;我们提供谷歌云、谷歌地图、谷歌办公套件、谷歌云认证培训服务。 开放表格式依赖嵌…

C++:string类模拟实现

C&#xff1a;string类模拟实现 成员变量构造和析构容量相关1.获取容器大小(_size)和容量(_capacity)2.扩容(reserve)3.更改容器大小 修改相关1.尾插2.指定位置插入3.指定位置删除4.清空5.交换两个对象 比较相关访问相关迭代器相关查找相关其它成员函数1.截取子串2.取得C格式字…

交融动画学习

学习抖音&#xff1a; 渡一前端教科频道 利用 filter 的属性实现交融效果 变成 让后利用这个效果实现一个功能 实现代码&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><style>* {margin: 0;…

c# .net mvc的IHttpHandler奇妙之旅--图片文件请求安全过滤,图片防盗链

在阅读该文章前,请先阅读该文章 c# .net mvc的IHttpHandler奇妙之旅。.net的生命周期和管道你听说过吗?你可以利用他处理业务如:跳转业务页面,文件请求的安全过滤,等等,还有许多神秘业务等着你去发现_橙-极纪元的博客-CSDN博客 该篇文章延续上面文章的第二小节《二、文件…

StarRocks企业级数据库

第1章 StarRocks简介 1.1 StarRocks介绍 StarRocks是新一代极速全场景MPP数据库 StraRocks充分吸收关系型OLAP数据库和分布式存储系统在大数据时代的优秀研究成果&#xff0c;在业界实践的基础上&#xff0c;进一步改进优化、升级架构&#xff0c;并增添了众多全新功能&…

Cadence OrCAD Capture CIS批量替换GND符号的方法

🏡《总目录》   🏡《宝典目录》 目录 1,概述2,方法3,总结1,概述 如下图所示,有时由于绘图是从多个地方复制粘贴而来,一个图纸中会存在多种GND符号。此时比较容易引起GND网络名不同意的问题,为了避免该问题可对其批量替换。 2,方法 第1步:选择需要替换的GND符号…