用 Java 解密 C# 加密的数据(DES)(转)
今天遇到java解密url的问题。我们的系统要获取外部传过来的URL,URL是采用 DES 算法对消息进行加密,再用 BASE64 编码。不过对方系统是用 C# 写的。
在网上搜了几篇文章终于找到一篇可以解决了,感谢这位老兄。
[原文地址:http://yilinliu.blogspot.com/2010/07/c-java-net-descryptoserviceprovider-vs.html]
前言: 這邊只列出我工作上有遇到的情況, 並未對所有的加解密做很完整的舉例/說明.
不過如果有人能提供其他情況, 我是可以試著去解看看.
情境: 在 .Net 中, 利用 DESCryptoServiceProvider 進行加密. 加密的 Mode 為 ECB, Padding 為 None, Zeros 與 PKCS7。
import javax.crypto.Cipher;
import javax.crypto.SecretKey;
import javax.crypto.SecretKeyFactory;
import javax.crypto.spec.DESKeySpec;
import org.apache.commons.codec.binary.Base64;
.Net 加密的程式如下: PaddingMode.None //建立一個Mode=ECB, Padding=None, Key為12345678的DESCryptoServiceProvider
DESCryptoServiceProvider objDESCryptoServiceProvider = new DESCryptoServiceProvider();
//Key的長度要64bits -> 8bytes;用ASCII編碼將Key轉為byte[]
objDESCryptoServiceProvider.Key = Encoding.ASCII.GetBytes("12345678");
objDESCryptoServiceProvider.Mode = CipherMode.ECB;
objDESCryptoServiceProvider.Padding = PaddingMode.None; //用UTF-8編碼, 將字串轉為byte[]
byte[] bysData = Encoding.UTF8.GetBytes("我是一個PaddingMode.None的測試字串!");
//因為PaddingMode.None的關係, byte[]的長度要是8的倍數
byte[] bysFixSizeData = new byte[(int)Math.Ceiling(bysData.Length / 8.0) * 8];
//將資料複製到長度為8的倍數的byte[]
Array.Copy(bysData, bysFixSizeData, bysData.Length); //進行加密
byte[] bysEncrypted = objDESCryptoServiceProvider.CreateEncryptor()
.TransformFinalBlock(bysFixSizeData, 0, bysFixSizeData.Length);
//將byte[]轉為Base64的字串
Console.WriteLine(Convert.ToBase64String(bysEncrypted));
//輸出: xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkfvLkf9phS9mW2tbtS6JcMSiwX2N1KbGp
Java 解密的程式如下:: 使用 org.apache.commons.codec.binary.Base64 這個類別做 Base64 字串的解碼.
try {
//解密的Key
String strKey = "12345678";
//已加密的Base64字串
String strEncrypted = "xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkfvLkf9phS9mW2tbtS6JcMSiwX2N1KbGp"; //先將Base64字串轉碼為byte[]
Base64 objBase64 = new Base64();
byte[] bysDecoded = objBase64.decode(strEncrypted.getBytes()); //建立解密所需的Key. 因為加密時的key是用ASCII轉換, 所以這邊也用ASCII做
DESKeySpec objDesKeySpec = new DESKeySpec(strKey.getBytes("ASCII"));
SecretKeyFactory objKeyFactory = SecretKeyFactory.getInstance("DES");
SecretKey objSecretKey = objKeyFactory.generateSecret(objDesKeySpec); //設定一個DES/ECB/NoPadding的Cipher
//ECB對應到.Net的CipherMode.ECB
//NoPadding對應到.Net的PaddingMode.None
Cipher objCipher = Cipher.getInstance("DES/ECB/NoPadding");
//設定為解密模式, 並設定解密的key
objCipher.init(Cipher.DECRYPT_MODE, objSecretKey); //輸出解密後的字串. 因為加密是用UTF-8將字串轉為byte[], 所以這邊要用UTF-8轉回去
//注意後面會多一些空字元. 因為加密前有因為資料長度的關係補上一些空的bytes
//這邊用String的trim()將這些空字元刪掉
String strDecrypted = new String(objCipher.doFinal(bysDecoded), "utf-8").trim();
System.out.println("[" + strDecrypted + "]");
//輸出:[我是一個PaddingMode.None的測試字串!]
} catch (Exception e) {
e.printStackTrace(System.out);
}
接下來測試一下 .Net 中的 PaddingMode.Zeros:
//建立一個Mode=ECB, Padding=None, Key為12345678的DESCryptoServiceProvider
DESCryptoServiceProvider objDESCryptoServiceProvider = new DESCryptoServiceProvider();
//Key的長度要64bits -> 8bytes
objDESCryptoServiceProvider.Key = Encoding.ASCII.GetBytes("12345678");
objDESCryptoServiceProvider.Mode = CipherMode.ECB;
objDESCryptoServiceProvider.Padding = PaddingMode.Zeros; //用UTF-8編碼, 將字串轉為byte[]
//因為PaddingMode.Zeros的關係, 不用像上一個Sample一樣, 另外弄一個byte[]去將資料補到8的倍數
byte[] bysData = Encoding.UTF8.GetBytes("我是一個PaddingMode.Zeros的測試字串!"); //進行加密
byte[] bysEncrypted = objDESCryptoServiceProvider.CreateEncryptor()
.TransformFinalBlock(bysData, 0, bysData.Length);
//將byte[]轉為Base64的字串
Console.WriteLine(Convert.ToBase64String(bysEncrypted));
//輸出: xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkYe6Sq5Y3mpq2JD91DLtxC/66itziI0rD
Java 的程式不用改變, 維持原狀即可.
try {
//解密的Key
String strKey = "12345678";
//已加密的Base64字串
String strEncrypted = "xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkfvLkf9phS9mW2tbtS6JcMSiwX2N1KbGp"; //先將Base64字串轉碼為byte[]
Base64 objBase64 = new Base64();
byte[] bysDecoded = objBase64.decode(strEncrypted.getBytes()); //建立解密所需的Key. 因為加密時的key是用ASCII轉換, 所以這邊也用ASCII做
DESKeySpec objDesKeySpec = new DESKeySpec(strKey.getBytes("ASCII"));
SecretKeyFactory objKeyFactory = SecretKeyFactory.getInstance("DES");
SecretKey objSecretKey = objKeyFactory.generateSecret(objDesKeySpec); //設定一個DES/ECB/NoPadding的Cipher
//ECB對應到.Net的CipherMode.ECB
//NoPadding對應到.Net的PaddingMode.None
Cipher objCipher = Cipher.getInstance("DES/ECB/NoPadding");
//設定為解密模式, 並設定解密的key
objCipher.init(Cipher.DECRYPT_MODE, objSecretKey); //輸出解密後的字串. 因為加密是用UTF-8將字串轉為byte[], 所以這邊要用UTF-8轉回去
//注意後面會多一些空字元. 因為加密前有因為資料長度的關係補上一些空的bytes
//這邊用String的trim()將這些空字元刪掉
String strDecrypted = new String(objCipher.doFinal(bysDecoded), "utf-8").trim();
System.out.println("[" + strDecrypted + "]");
//輸出:[我是一個PaddingMode.None的測試字串!]
} catch (Exception e) {
e.printStackTrace(System.out);
}
最後測試 .Net 中的 PaddingMode.PKCS7:
//建立一個Mode=ECB, Padding=None, Key為12345678的DESCryptoServiceProvider
DESCryptoServiceProvider objDESCryptoServiceProvider = new DESCryptoServiceProvider();
//Key的長度要64bits -> 8bytes
objDESCryptoServiceProvider.Key = Encoding.ASCII.GetBytes("12345678");
objDESCryptoServiceProvider.Mode = CipherMode.ECB;
objDESCryptoServiceProvider.Padding = PaddingMode.PKCS7; //用UTF-8編碼, 將字串轉為byte[]
//這邊也不用另外做補資料的動作
byte[] bysData = Encoding.UTF8.GetBytes("我是一個PaddingMode.PKCS7的測試字串!"); //進行加密
byte[] bysEncrypted = objDESCryptoServiceProvider.CreateEncryptor()
.TransformFinalBlock(bysData, 0, bysData.Length);
//將byte[]轉為Base64的字串
Console.WriteLine(Convert.ToBase64String(bysEncrypted));
//輸出: xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkt86hRLUM4/m2JD91DLtxC5+8Tqc7iB2f
Java 程式的差別就在於把 DES / ECB / NoPadding 改為 DES / ECB / PKCS5Padding.
try {
//解密的Key
String strKey = "12345678";
//已加密的Base64字串
String strEncrypted = "xnygZZ+WkN4pmDVDjBJ41o9LePFibMJkt86hRLUM4/m2JD91DLtxC5+8Tqc7iB2f"; //先將Base64字串轉碼為byte[]
Base64 objBase64 = new Base64();
byte[] bysDecoded = objBase64.decode(strEncrypted.getBytes()); //建立解密所需的Key. 因為加密時的key是用ASCII轉換, 所以這邊也用ASCII做
DESKeySpec objDesKeySpec = new DESKeySpec(strKey.getBytes("ASCII"));
SecretKeyFactory objKeyFactory = SecretKeyFactory.getInstance("DES");
SecretKey objSecretKey = objKeyFactory.generateSecret(objDesKeySpec); //設定一個DES/ECB/PKCS5Padding的Cipher
//ECB對應到.Net的CipherMode.ECB
//用PKCS5Padding對應到.Net的PaddingMode.PKCS7
Cipher objCipher = Cipher.getInstance("DES/ECB/PKCS5Padding");
//設定為解密模式, 並設定解密的key
objCipher.init(Cipher.DECRYPT_MODE, objSecretKey); //輸出解密後的字串. 因為加密時指定PaddingMode.PKCS7, 所以可以不用處理空字元
//不過若想保險點, 也是可以用trim()去處理過一遍
String strDecrypted = new String(objCipher.doFinal(bysDecoded), "utf-8").trim();
System.out.println("[" + strDecrypted + "]");
//輸出:[我是一個PaddingMode.PKCS7的測試字串!]
} catch (Exception e) {
e.printStackTrace(System.out);
}
用 Java 解密 C# 加密的数据(DES)(转)的更多相关文章
- c#加密,java解密(3DES加密)
c#代码 using System; using System.Security; using System.Security.Cryptography; using System.IO; using ...
- 【知识积累】DES算法之C#加密&Java解密
一.前言 在项目需要添加安全模块,客户端调用服务端发布的service必须要经过验证,加密算法采用DES,客户端采用C#进行加密,服务端使用Java进行解密.废话不多说,直接上代码. 二.客户端 客户 ...
- Java Des加解密方法(c#加密Java解密)
最近我们用Java把一个用.net编写的老系统重新做了翻版,但是登录还是用.net的登录.这样就会遇到一个比较棘手的问题,我们登录用的cookie信息都是.net用des加密的,但我们不得不用Java ...
- PHP DES解密 对应Java SHA1PRNG方式加密
背景及问题 背景:在和外部系统通过HTTP方式跳转时, 为保障传输参数安全性, 采用AES 加密参数. 关于对称加密中 AES, DES, CBC, ECB, PKCS5Padding 概念可参考ht ...
- 利用DES,C#加密,Java解密代码
//C#加密 /// <summary> /// 进行DES加密. /// </summary> /// <param name="pToEncrypt&quo ...
- DES c#加密后java解密
public static String byteArr2HexStr(byte[] bytIn) { StringBuilder builder = new StringBuilder(); for ...
- Java中3DES加密解密与其他语言(如C/C++)通信
国内私募机构九鼎控股打造APP,来就送 20元现金领取地址:http://jdb.jiudingcapital.com/phone.html内部邀请码:C8E245J (不写邀请码,没有现金送)国内私 ...
- 对称加密----AES和DES加密、解密
目前主流的加密方式有:(对称加密)AES.DES (非对称加密)RSA.DSA 调用AES/DES加密算法包最精要的就是下面两句话: Cipher cipher = Cipher.get ...
- Java常用的加密解密类(对称加密类)
Java常用的加密解密类 原文转载至:http://blog.csdn.net/wyc_cs/article/details/8793198 原创 2013年04月12日 14:33:35 1704 ...
随机推荐
- C# 操作Session、Cookie,Url 编码解码工具类WebHelper
using System; using System.Collections.Generic; using System.IO; using System.Net; using System.Text ...
- Qt-c++桌面编程报错:qt.qpa.plugin: Could not find the Qt platform plugin "windows" in "",已解决
语言:c++ 编译库:Qt GUI,qt5.12.1 软件类型:Qt application,qt桌面软件 运行平台:window 10 ?按照[https://www.devbean.net/201 ...
- Python3学习笔记之十九
1. 什么是orm? object relation mapping 对象关系映射 一旦确定表关系为一对多:在多的表中添加关联字段. 一对一:可以在任意一张表添加关联字段. 多对多:创建第三 ...
- curl 查看一个web站点的响应时间
1. curl 查看web站点 curl -o /dev/null -s -w "time_namelookup:%{time_namelookup}s\ntime_connect:%{ti ...
- jq判断是PC还是手机端的方法
$(function(){ //判断是否是手机 var mobile_flag = isMobile(); if(mobile_flag){ $('.now_qq').attr('href',&quo ...
- FbinstTools制作多系统启动U盘(Windows+Linux)
U盘启动盘制作工具在国内有倆工具,老毛桃.大白菜.也不知道是谁模仿谁的,反正PE肯定是Microsoft的. PE其实就是精简版的Windows维护系统,那如何制作Linux启动盘呢,百度搜“linu ...
- Virtual box中Ubuntu虚拟机磁盘碎片整理和空间清理方法
虚拟机中,随着不断的使用,增加大文件(例如日志,视频和软件版本),虽然在虚拟机中手动删除了,但是虚拟机占用的空间并不会随之减少,需要手动清理一下. 这里介绍一种Virtual box中Ubuntu碎片 ...
- Github 上怎样把新 commits 使用在自己的 fork 上
作者:黄晓佳 链接:https://www.zhihu.com/question/20393785/answer/105370502 来源:知乎 著作权归作者所有.商业转载请联系作者获得授权,非商业转 ...
- SSH(Struts+spring+hibernate)配置
1.spring和struts 1)web.xml 配置spring的ContextLoaderListener(监听器) 配置Struts的StrutsPrepareAndExecuteFilter ...
- css边框动画
<div class="button">hover me to change</div> .button{ width:200px; height:60px ...