C# 3DES加密详解
作者:hebedich 发布时间:2022-12-01 05:58:55
标签:c#,3des
最近一个项目中,因为服务端是用的java开发的,客户端是用的C#,由于通信部分采用到了3DES加密,所以做个记录,以备以后需要的时候直接用。
这是对方(java)的加密算法,和网上流传的代码也差不多(主密钥直接写死了,方便测试)
package org.zwork.market.mina.msg;
import java.security.spec.KeySpec;
import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESKeySpec;
import javax.crypto.spec.DESedeKeySpec;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.zwork.market.MktContants;
public class ThreeEncryptDecrypt {
private static final Logger LOGGER = LoggerFactory.getLogger(ThreeEncryptDecrypt.class);
// 定义 加密算法,可用 DES,DESede,Blowfish
public static final String Algorithm = "DESede";
public static String DES = "DES/ECB/NoPadding";
public static String TriDes = "DESede/ECB/NoPadding";
// des加密
public static byte[] des_crypt(byte key[], byte data[]) {
try {
KeySpec ks = new DESKeySpec(key);
SecretKeyFactory kf = SecretKeyFactory.getInstance("DES");
SecretKey ky = kf.generateSecret(ks);
Cipher c = Cipher.getInstance(DES);
c.init(Cipher.ENCRYPT_MODE, ky);
return c.doFinal(data);
} catch (Exception e) {
LOGGER.error("des_crypt error:", e);
return null;
}
}
// des解密
public static byte[] des_decrypt(byte key[], byte data[]) {
try {
KeySpec ks = new DESKeySpec(key);
SecretKeyFactory kf = SecretKeyFactory.getInstance("DES");
SecretKey ky = kf.generateSecret(ks);
Cipher c = Cipher.getInstance(DES);
c.init(Cipher.DECRYPT_MODE, ky);
return c.doFinal(data);
} catch (Exception e) {
LOGGER.error("des_decrypt error:", e);
return null;
}
}
// 3DES加密
public static byte[] trides_crypt(byte key[], byte data[]) {
try {
byte[] k = new byte[24];
int len = data.length;
if (data.length % 8 != 0) {
len = data.length - data.length % 8 + 8;
}
byte[] needData = null;
if (len != 0)
needData = new byte[len];
for (int i = 0; i < len; i++) {
needData[i] = 0x00;
}
System.arraycopy(data, 0, needData, 0, data.length);
if (key.length == 16) {
System.arraycopy(key, 0, k, 0, key.length);
System.arraycopy(key, 0, k, 16, 8);
} else {
System.arraycopy(key, 0, k, 0, 24);
}
KeySpec ks = new DESedeKeySpec(k);
SecretKeyFactory kf = SecretKeyFactory.getInstance("DESede");
SecretKey ky = kf.generateSecret(ks);
Cipher c = Cipher.getInstance(TriDes);
c.init(Cipher.ENCRYPT_MODE, ky);
return c.doFinal(needData);
} catch (Exception e) {
LOGGER.error("trides_crypt error:", e);
return null;
}
}
// 3DES解密
public static byte[] trides_decrypt(byte key[], byte data[]) {
try {
byte[] k = new byte[24];
int len = data.length;
if (data.length % 8 != 0) {
len = data.length - data.length % 8 + 8;
}
byte[] needData = null;
if (len != 0)
needData = new byte[len];
for (int i = 0; i < len; i++) {
needData[i] = 0x00;
}
System.arraycopy(data, 0, needData, 0, data.length);
if (key.length == 16) {
System.arraycopy(key, 0, k, 0, key.length);
System.arraycopy(key, 0, k, 16, 8);
} else {
System.arraycopy(key, 0, k, 0, 24);
}
KeySpec ks = new DESedeKeySpec(k);
SecretKeyFactory kf = SecretKeyFactory.getInstance("DESede");
SecretKey ky = kf.generateSecret(ks);
Cipher c = Cipher.getInstance(TriDes);
c.init(Cipher.DECRYPT_MODE, ky);
return c.doFinal(needData);
} catch (Exception e) {
LOGGER.error("trides_decrypt error:", e);
return null;
}
}
public static String getPass(String source) {
byte[] data= hexToBytes(source);
byte[] key ="111111111111111111111111111a1.1.".getBytes();
String result = byte2hex(trides_decrypt(key, data)).toUpperCase();
return result.substring(2, 8);
}
public static String byte2hex(byte[] data) {
StringBuffer sb = new StringBuffer();
for (int i = 0; i < data.length; i++) {
String temp = Integer.toHexString(((int) data[i]) & 0xFF);
for (int t = temp.length(); t < 2; t++) {
sb.append("0");
}
sb.append(temp);
}
return sb.toString();
}
public static byte[] hexToBytes(String str) {
if (str == null) {
return null;
} else if (str.length() < 2) {
return null;
} else {
int len = str.length() / 2;
byte[] buffer = new byte[len];
for (int i = 0; i < len; i++) {
buffer[i] = (byte) Integer.parseInt(str.substring(i * 2, i * 2 + 2), 16);
}
return buffer;
}
}
}
因为客户端只负责数据加密,所以我这里只写了加密的部分。由于java和C#语言很相似,所以我就仿这他们给的java代码改成C#的,当然也在网上参考了一些代码,不过中间还是出现了些问题,比如C#不支持弱密钥(把密钥弄复杂点),因为没注意大小写造成加密结果不一致等等。编程这东西一个点都能让整个系统崩溃,所以小细节很重要!!!
public class DESHelper
{
/// <summary>
/// 将密码转成直接数组
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static byte[] HexToBytes(String str)
{
if (str == null)
{
return null;
}
else if (str.Length < 2)
{
return null;
}
else
{
int len = str.Length / 2;
byte[] buffer = new byte[len];
for (int i = 0; i < len; i++)
{
var temp = str.Substring(i * 2, 2);
buffer[i] = (byte)Convert.ToInt32(temp, 16);
}
return buffer;
}
}
/// <summary>
/// 3DES加密
/// </summary>
/// <param name="key"></param>
/// <param name="data"></param>
/// <returns></returns>
public static byte[] GetDes3EncryptedText(byte[] key, byte[] data)
{
byte[] k = new byte[24];
int len = data.Length;
if (data.Length % 8 != 0)
{
len = data.Length - data.Length % 8 + 8;
}
byte[] needData = null;
if (len != 0)
needData = new byte[len];
for (int i = 0; i < len; i++)
{
needData[i] = 0x00;
}
Buffer.BlockCopy(data, 0, needData, 0, data.Length);
if (key.Length == 16)
{
Buffer.BlockCopy(key, 0, k, 0, key.Length);
Buffer.BlockCopy(key, 0, k, 16, 8);
}
else
{
Buffer.BlockCopy(key, 0, k, 0, 24);
}
var des3 = new TripleDESCryptoServiceProvider();
des3.Key = k;
des3.Mode = CipherMode.ECB;
des3.Padding = PaddingMode.Zeros;
using (MemoryStream ms = new MemoryStream())
using (CryptoStream cs = new CryptoStream(ms, des3.CreateEncryptor(), CryptoStreamMode.Write))
{
cs.Write(data, 0, data.Length);
cs.FlushFinalBlock();
return ms.ToArray();
}
}
/// <summary>
/// 将加密结果转成字符串
/// </summary>
/// <param name="data"></param>
/// <returns></returns>
public static String GetByte2Hex(byte[] data)
{
StringBuilder sb = new StringBuilder();
for (int i = 0; i < data.Length; i++)
{
String temp = string.Format("{0:X}", ((int)data[i]) & 0xFF);
for (int t = temp.Length; t < 2; t++)
{
sb.Append("0");
}
sb.Append(temp);
}
return sb.ToString();
}
}
以上所述就是本文的全部内容了,希望大家能够喜欢。


猜你喜欢
- C#事件标准命名规则一些开源代码的事件命名很混乱,以此文章用作本人以后工作的参考。事件的名称事件始终是指某个操作,这个操作可能正在发生,也可
- where 子句用于指定类型约束,这些约束可以作为泛型声明中定义的类型参数的变量。1.接口约束。例如,可以声明一个泛型类 MyGeneric
- Java 定时器在JAVA中实现定时器功能要用的二个类是Timer,TimerTaskTimer类是用来执行任务的类,它接受一个
- -vmargs -Xms128M -Xmx512M -XX:PermSize=64M -XX:MaxPermSize=128M 这里有几个问
- 有小伙伴表示微人事(https://github.com/lenve/vhr)的权限粒度不够细。不过松哥想说的是,技术都是相通的,明白了 v
- 本文实例讲述了Android开发之App widget用法。分享给大家供大家参考,具体如下:放在桌面上的控件叫做——App widget,例
- 实例如下://选择文件,点击【浏览】,选择文件 private void button1_Click(object sender, Even
- 本文实例为大家分享了java实现录音播放的具体代码,供大家参考,具体内容如下需求:1.实现可以从麦克风进行录音2.可以停止录音3.实现播放录
- Java IO 转化流乱码引起转换流读取乱码读取电脑磁盘上的Java.txt文件内容,文件路径: e:\Java\Java.txt
- /* * Copyright 2012-2013 The Haohui Network Cor
- Android 中View.onDraw(Canvas canvas)的使用方法View通过View.onDraw(Canvas canva
- 知识准备Timer和ScheduledExecutorService是JDK内置的定时任务方案,而业内还有一个经典的定时任务的设计叫时间轮(
- 为大家分享的解决MyEclipse中的Building workspace问题的方法如下方法一:点击“Project”,取消勾选“Build
- 这里介绍一个简易的音乐播放器,供大家参考,具体内容如下效果图如下:但是,由于这是一个简易版的音乐播放器,所播放的音乐只有一首,且被写死,但,
- springboot service内组件加载顺序先加载自身构造器,所以在构造器中初始化时若使用需要注入的(即@Autowired注解的)组
- 前言在逛网易新闻时,发现列表中的广告在你滑动的时候会有一个3D旋转的交互引你的注意,不得不说这些产品为了让用户看广告花样百出,那么今天我们就
- 嗯,就是BASE64,不用多想,本来计划是要跟上一篇字符串压缩一起写的,用来实现将一个文件可以用json或者text等方式进行接口之间的传输
- 定义可理解为 适配广泛的类型,即参数化类型,可以把类型像方法的参数那样进行传递。// 以ArrayList为示例// 泛型T可以是任意类pu
- 如何下载并配置JDK 15进入官网下载JDK 15。官网地址:https://www.oracle.com/index.html脚本之家下载
- 说明本实例能够监控聚划算的抢购按钮,在聚划算整点聚的时间到达时自动弹开页面(URL自己定义)。可以自定义监控持续分钟数,同时还可以通过多线程