.NET和java的RSA互通,仅此而已
.NET和java的RSA互通,仅此而已
在开始这篇文章之前,先请读者朋友阅读老唐的这两篇文章:
1、Java与.Net环境下RSA加密解密交互不成功的问题解决
2、Java与.Net环境下RSA加密解密交互不成功的问题解决【续】
和我的这篇文章
前面老唐的两篇文章中提到,要想实现.NET和Java的RSA互通,只能抛弃.NET现有的加密算法,而是利用http://www.codeproject.com/csharp/biginteger.asp 项目中的BigInteger类(.NET Framework4中已增加了这个类的实现,在System.Numberic命名空间中),这个BigInteger类实际上就是仿照着java的BigInteger类来写的。
利用这个类的确可以很好的实现RSA的加解密,比如,在.NET端,构建一个公钥对应的BigInteger e、一个模对应的BigInteger n和一个明文对应的BigInteger m,然后执行语句BigInteger c=m.modPow(e,n),便可以实现加密操作,密文为c,这样的加密是标准加密,没有附加任何填充算法的加密。
老唐的文章中说,不能互通是因为加密标准不一样,导致一方加密而另一方不能解密,其实不然,.NET采用的加密标准是PKCS1Padding(或OAEPPadding——只支持XP以上版本),这也是我在前面一篇文章中提到的一种填充算法,而java同样支持这一填充标准,既然可以遵循统一的标准,那么.NET和java的RSA互通,无需添加任何新代码便可以轻松实现!
请看下面的示例(.NET端加密,Java端解密):
Java端代码:
- import java.math.BigInteger;
- import java.util.Scanner;
- import java.security.KeyFactory;
- import java.security.PrivateKey;
- import java.security.KeyPair;
- import java.security.KeyPairGenerator;
- import java.security.PublicKey;
- import java.security.interfaces.RSAPrivateKey;
- import java.security.interfaces.RSAPublicKey;
- import java.security.spec.RSAPublicKeySpec;
- import javax.crypto.Cipher;
- import sun.misc.*;
- public class RsaKey {
- public static void main(String[] args) throws Exception {
- //生成公私钥对
- KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA");
- keyPairGen.initialize(1024);
- KeyPair keyPair = keyPairGen.generateKeyPair();
- PublicKey publicKey = (RSAPublicKey) keyPair.getPublic();
- PrivateKey privateKey = (RSAPrivateKey) keyPair.getPrivate();
- //将公钥和模进行Base64编码
- KeyFactory keyFactory = KeyFactory.getInstance("RSA");
- RSAPublicKeySpec publicSpec= keyFactory.getKeySpec(publicKey,RSAPublicKeySpec.class);
- BigInteger modulus = publicSpec.getModulus();
- BigInteger exponent=publicSpec.getPublicExponent();
- byte[] ary_m=modulus.toByteArray();//注意:对公钥和模进行Base64编码时,不是对BigInteger对应的字符串编码,而是对其内部 的字节数组进行编码
- byte[] ary_e=exponent.toByteArray();
- String str_m;
- String str_e;
- if(ary_m[0]==0 && ary_m.length==129)//判断数组首元素是否为0,若是,则将其删除,保证模的位数是128
- {
- byte[] temp=new byte[ary_m.length-1];
- for(int i=1;i<ary_m.length;i++)
- {
- temp[i-1]=ary_m[i];
- }
- str_m=(new BASE64Encoder()).encodeBuffer(temp);
- }
- else
- {
- str_m=(new BASE64Encoder()).encodeBuffer(ary_m);
- }
- str_e=(new BASE64Encoder()).encodeBuffer(ary_e);
- System.out.println("公钥为:"+str_e);
- System.out.println("模为:"+str_m);
- System.out.println("运行.NET程序,用所提供的公钥和模进行加密,然后将加密结果输入本程序进行解密:");
- Scanner sc=new Scanner(System.in);
- String str_en="";
- String st="";
- while(!(st=sc.nextLine()).equals(""))
- {
- str_en+=st;
- }
- byte[] ary_en=(new BASE64Decoder()).decodeBuffer(str_en);
- //解密
- //注意Cipher初始化时的参数“RSA/ECB/PKCS1Padding”,代表和.NET用相同的填充算法,如果是标准RSA加密,则参数为“RSA”
- Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1Padding");
- cipher.init(Cipher.DECRYPT_MODE, privateKey);
- byte[] deBytes = cipher.doFinal(ary_en);
- String s = new String(deBytes );
- System.out.println("解密结果为:" + s);
- }
- }
Java端演示截图
.NET端代码:
- static void Main(string[] args)
- {
- try
- {
- RSACryptoServiceProvider rsa = new RSACryptoServiceProvider();
- RSAParameters para = new RSAParameters();
- //加密
- Console.WriteLine("请输入公钥");
- string publicKey = Console.ReadLine();
- Console.WriteLine("请输入模:");
- string modulus = Console.ReadLine();
- while(true)
- {
- string s = Console.ReadLine();
- if (s == "")
- {
- break;
- }
- else
- {
- modulus += s;
- }
- }
- Console.WriteLine("请输入明文:");
- string m = Console.ReadLine();
- para.Exponent = Convert.FromBase64String(publicKey);
- para.Modulus = Convert.FromBase64String(modulus);
- rsa.ImportParameters(para);
- byte[] enBytes = rsa.Encrypt(UTF8Encoding.UTF8.GetBytes(m),false);
- Console.WriteLine("密文为:"+Convert.ToBase64String(enBytes));
- Console.ReadLine();
- }
- catch(Exception ex)
- {
- Console.WriteLine(ex.Message);
- Console.ReadLine();
- }
.NET端演示截图:
接下来的示例是(java端加密,.NET端解密):
.net端代码:
- static void Main(string[] args)
- {
- try
- {
- RSACryptoServiceProvider rsa = new RSACryptoServiceProvider();
- RSAParameters para = rsa.ExportParameters(true);
- //加¨®密¨¹
- Console.WriteLine("公?钥?为a:êo"+ Convert.ToBase64String(para.Exponent));
- Console.WriteLine("模¡ê为a:êo" + Convert.ToBase64String(para.Modulus));
- Console.WriteLine("请?输º?入¨?密¨¹文?");
- string enStr = Console.ReadLine();
- while(true)
- {
- string s = Console.ReadLine();
- if (s == "")
- {
- break;
- }
- else
- {
- enStr += s;
- }
- }
- byte[] deBytes = rsa.Decrypt(Convert.FromBase64String(enStr),false);
- Console.WriteLine("明¡Â文?为a:êo"+UTF8Encoding.UTF8.GetString(deBytes));
- Console.ReadLine();
- }
- catch(Exception ex)
- {
- Console.WriteLine(ex.Message);
- Console.ReadLine();
- }
- }
Java端代码:
- public static void main(String[] args) throws Exception {
- Scanner sc=new Scanner(System.in);
- //获取公钥、模及明文的字符串
- System.out.println("请输入公钥:");
- String str_exponent=sc.nextLine();
- System.out.println("请输入模:");
- String str_modulus="";
- String st="";
- while(!(st=sc.nextLine()).equals(""))
- {
- str_modulus+=st;
- }
- System.out.println("请输入明文:");
- String str_m=sc.nextLine();
- //创建公钥
- byte[] ary_exponent=(new BASE64Decoder()).decodeBuffer(str_exponent);
- byte[] ary_modulus=(new BASE64Decoder()).decodeBuffer(str_modulus);
- //注意构造函数,调用时指明正负值,1代表正值,否则报错
- BigInteger big_exponent = new BigInteger(1,ary_exponent);
- BigInteger big_modulus = new BigInteger(1,ary_modulus);
- RSAPublicKeySpec keyspec=new RSAPublicKeySpec(big_modulus,big_exponent);
- KeyFactory keyfac=KeyFactory.getInstance("RSA");
- PublicKey publicKey=keyfac.generatePublic(keyspec);
- //进行加密
- Cipher cipher = Cipher.getInstance("RSA/ECB/PKCS1Padding");
- cipher.init(Cipher.ENCRYPT_MODE, publicKey);
- byte[] enBytes = cipher.doFinal(str_m.getBytes());
- String s = (new BASE64Encoder()).encodeBuffer(enBytes);
- System.out.println("加密结果为:" + s);
- }
.NET和java的RSA互通,仅此而已的更多相关文章
- Java & PHP RSA 互通密钥、签名、验签、加密、解密
RSA加密算法是一种非对称加密算法.在公开密钥加密和电子商业中RSA被广泛使用.RSA是1977年由罗纳德·李维斯特(Ron Rivest).阿迪·萨莫尔(Adi Shamir)和伦纳德·阿德曼(Le ...
- php与JAVA的RSA加密互通
Java 版本RSA 进行加密解密 在网上查询了好几天,最终找到解决方案,网络上都是通过Cipher.getInstance("RSA"); 而改成Cipher.getInstan ...
- Java加密算法 RSA
Java加密算法 RSA 2015-06-06 08:44 511人阅读 评论(0) 收藏 举报 分类: JAVA(57) 公钥加密也称为非对称加密.速度慢.加密和解密的钥匙不相同,某一个人持有私 ...
- Java前端Rsa公钥加密,后端Rsa私钥解密(目前还不支持中文加密解密,其他都行)
Base64工具类,可以让rsa编码的乱码变成一串字符序列 package com.utils; import java.io.ByteArrayInputStream; import java.io ...
- Java前端Rsa公钥加密,后端Rsa私钥解密(支持字符和中文)
Base64工具类,可以让rsa编码的乱码变成一串字符序列 package com.utils; import java.io.ByteArrayInputStream; import java.io ...
- JAVA实现RSA加密,非对称加密算法
RSA.java package org.icesnow.jeasywx.util.security; import java.security.Key; import java.security.K ...
- JAVA实现RSA加密解密 非对称算法
首先RSA是一个非对称的加密算法.所以在使用该算法加密解密之前,必须先行生成密钥对.包含公钥和私钥 JDK中提供了生成密钥对的类KeyPairGenerator,实比例如以下: public stat ...
- 【转】 Java 进行 RSA 加解密时不得不考虑到的那些事儿
[转] Java 进行 RSA 加解密时不得不考虑到的那些事儿 1. 加密的系统不要具备解密的功能,否则 RSA 可能不太合适 公钥加密,私钥解密.加密的系统和解密的系统分开部署,加密的系统不应该同时 ...
- java生成RSA公私钥字符串,简单易懂
java生成RSA公私钥字符串,简单易懂 解决方法: 1.下载bcprov-jdk16-140.jar包,参考:http://www.yayihouse.com/yayishuwu/chapter ...
随机推荐
- ODBC接口规范
第1章 ODBC API基础 1.1 ODBC API句柄 ODBC API 实现数据库操作的手段是句柄.在ODBC中,使用不同的句柄(HANDLE)来标志环境(environment).连接 ...
- Cache
1.Cache中的块与主存储器中的块时按照什么样的规则建立对应关系的? 2.在这种对应关系下,主存地址又是如何变换成Cache地址的? Cache信息: 1.数据Cache和指令Cache是分开还是统 ...
- C语言 负数取余的原理
负数求余数运算是一个数学问题: 任何一个整数n都可以表示成 n=k*q+r 其中0<=|r|<|q| 这里的r就是n除以q的余数,即 r==n%q 例如: -9=(-2)*4+(-1) 则 ...
- flexbox的术语
在详细阅读这篇文章之前,我们很有必要先了解flexbox的几个常用术语,这样有助于大家对后文的理解. 伸缩容器:一个设有“display:flex”或“display:inline-flex”的元素 ...
- ResponsiveSlides.js最轻量级的幻灯片插件
摘要:ResponsiveSlides.js是一个展示同一容器内图片的轻量级响应式jQuery幻灯片插件它支持包括IE6在内的几乎所有的浏览器,在IE6中还支持最大宽度属性,但在其它浏览器中并不原生支 ...
- 4.2springmvc校验
1.hibernate的校验框架validation所需要jar包: 2 在applicationContext.xml中配置校验器: <!-- 校验器 --> <bean id=& ...
- C#部分---特殊集合:stack栈集合、queue队列集合、哈希表集合。
1.stack栈集合:又名 干草堆集合 栈集合 特点:(1)一个一个赋值 一个一个取值(2)先进后出实例化 初始化 Stack st = new Stack(); //添加元素用push st.Pus ...
- [poj 3261]Milk Patterns
后缀数组搞一下就可以了喵~ 其实这道题的第一个想法是 SAM ,建完后缀自动机后拓扑排序跑一遍统计下每个子串的出现次数就 O(N) 就妥妥过掉了 后缀树也是 O(N) 的,统计一下每个节点对应的子树中 ...
- (转)The 9 Deep Learning Papers You Need To Know About (Understanding CNNs Part 3)
Adit Deshpande CS Undergrad at UCLA ('19) Blog About The 9 Deep Learning Papers You Need To Know Abo ...
- 论文笔记之:Learning Multi-Domain Convolutional Neural Networks for Visual Tracking
Learning Multi-Domain Convolutional Neural Networks for Visual Tracking CVPR 2016 本文提出了一种新的CNN 框架来处理 ...