- 浏览: 2830447 次
- 性别:
- 来自: 武汉
文章分类
- 全部博客 (1173)
- 名言警句 (5)
- 心情随笔 (50)
- 数据库 (57)
- Java基础 (241)
- J2EE框架 (91)
- 数据结构 (12)
- 程序设计 (21)
- WEB技术 (128)
- 网络日志 (12)
- IT资讯 (247)
- linux (64)
- solaris (2)
- 其它 (143)
- WebService (4)
- 日语学习 (2)
- 机器人 (5)
- Android (5)
- cgywin (3)
- Game (1)
- DWR (1)
- spring (8)
- canvas (1)
- Guava (3)
- Modbus (5)
- 测试 (6)
- mongodb (9)
- Quartz (2)
- Cron (1)
- windows (2)
- 持续集成 (1)
- bootstrap (3)
- 结对编程 (1)
- nodejs (1)
- Netty (1)
- 安全 (3)
- webstorm (2)
- sparkline (1)
- Job (1)
- git (3)
- Maven (3)
- knockout (5)
- jquery (1)
- bower (1)
- docker (1)
- confluence (4)
- wiki (1)
- GoogleMap (1)
- jekyll (10)
- ruby (2)
- npm (3)
- browserify (1)
- gulp (3)
- openwrt (1)
- discuz (3)
- 输入法 (1)
- JPA (1)
- eclipse (2)
- IntelliJ (1)
- css (1)
- 虚拟机 (1)
- 操作系统 (1)
- azkaban (2)
- scrum (1)
最新评论
-
pangxiea_:
你好, 想请问一下 Linux下 这么使用rxtxcomm 在 ...
使用Java进行串口通信 -
abababudei:
请教一下,这个您是怎么解决的:/dev/ttyS2enteri ...
Java应用程序的MODBUS通讯 -
xuniverse:
hannibal005 写道楼主,我问下 request.se ...
用javascript与java进行RSA加密与解密 -
atxkm:
找了一下午,终于找到了
gulp 拷贝文件时如何移除文件目录结构 -
kalogen:
gtczr 写道非常感谢,经过我自己的修改,已经完美实现。发出 ...
用javascript与java进行RSA加密与解密
这几天一直做安全登录,网上查了好多资料,不尽如意。
具体实现思路如下:
1。服务端生成公钥与私钥,保存。
2。客户端在请求到登录页面后,随机生成一字符串。
3。后此随机字符串作为密钥加密密码,再用从服务端获取到的公钥加密生成的随机字符串。
4。将此两段密文传入服务端,服务端用私钥解出随机字符串,再用此私钥解出加密的密文。
这其中有一个关键是解决服务端的公钥,传入客户端,客户端用此公钥加密字符串后,后又能在服务端用私钥解出。
此文即为实现此步而作。
加密算法为RSA:
1。服务端的RSA java实现。
/** * */ package com.sunsoft.struts.util; import java.io.ByteArrayOutputStream; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.math.BigInteger; import java.security.KeyFactory; import java.security.KeyPair; import java.security.KeyPairGenerator; import java.security.NoSuchAlgorithmException; import java.security.PrivateKey; import java.security.PublicKey; import java.security.SecureRandom; import java.security.interfaces.RSAPrivateKey; import java.security.interfaces.RSAPublicKey; import java.security.spec.InvalidKeySpecException; import java.security.spec.RSAPrivateKeySpec; import java.security.spec.RSAPublicKeySpec; import javax.crypto.Cipher; /** * RSA 工具类。提供加密,解密,生成密钥对等方法。 * 需要到http://www.bouncycastle.org下载bcprov-jdk14-123.jar。 * */ public class RSAUtil { /** * * 生成密钥对 * * * @return KeyPair * * @throws EncryptException */ public static KeyPair generateKeyPair() throws Exception { try { KeyPairGenerator keyPairGen = KeyPairGenerator.getInstance("RSA", new org.bouncycastle.jce.provider.BouncyCastleProvider()); final int KEY_SIZE = 1024;// 没什么好说的了,这个值关系到块加密的大小,可以更改,但是不要太大,否则效率会低 keyPairGen.initialize(KEY_SIZE, new SecureRandom()); KeyPair keyPair = keyPairGen.generateKeyPair(); saveKeyPair(keyPair); return keyPair; } catch (Exception e) { throw new Exception(e.getMessage()); } } public static KeyPair getKeyPair()throws Exception{ FileInputStream fis = new FileInputStream("C:/RSAKey.txt"); ObjectInputStream oos = new ObjectInputStream(fis); KeyPair kp= (KeyPair) oos.readObject(); oos.close(); fis.close(); return kp; } public static void saveKeyPair(KeyPair kp)throws Exception{ FileOutputStream fos = new FileOutputStream("C:/RSAKey.txt"); ObjectOutputStream oos = new ObjectOutputStream(fos); //生成密钥 oos.writeObject(kp); oos.close(); fos.close(); } /** * * 生成公钥 * * * @param modulus * * @param publicExponent * * @return RSAPublicKey * * @throws Exception */ public static RSAPublicKey generateRSAPublicKey(byte[] modulus, byte[] publicExponent) throws Exception { KeyFactory keyFac = null; try { keyFac = KeyFactory.getInstance("RSA", new org.bouncycastle.jce.provider.BouncyCastleProvider()); } catch (NoSuchAlgorithmException ex) { throw new Exception(ex.getMessage()); } RSAPublicKeySpec pubKeySpec = new RSAPublicKeySpec(new BigInteger( modulus), new BigInteger(publicExponent)); try { return (RSAPublicKey) keyFac.generatePublic(pubKeySpec); } catch (InvalidKeySpecException ex) { throw new Exception(ex.getMessage()); } } /** * * 生成私钥 * * * @param modulus * * @param privateExponent * * @return RSAPrivateKey * * @throws Exception */ public static RSAPrivateKey generateRSAPrivateKey(byte[] modulus, byte[] privateExponent) throws Exception { KeyFactory keyFac = null; try { keyFac = KeyFactory.getInstance("RSA", new org.bouncycastle.jce.provider.BouncyCastleProvider()); } catch (NoSuchAlgorithmException ex) { throw new Exception(ex.getMessage()); } RSAPrivateKeySpec priKeySpec = new RSAPrivateKeySpec(new BigInteger( modulus), new BigInteger(privateExponent)); try { return (RSAPrivateKey) keyFac.generatePrivate(priKeySpec); } catch (InvalidKeySpecException ex) { throw new Exception(ex.getMessage()); } } /** * * 加密 * * * @param key * 加密的密钥 * * @param data * 待加密的明文数据 * * @return 加密后的数据 * * @throws Exception */ public static byte[] encrypt(PublicKey pk, byte[] data) throws Exception { try { Cipher cipher = Cipher.getInstance("RSA", new org.bouncycastle.jce.provider.BouncyCastleProvider()); cipher.init(Cipher.ENCRYPT_MODE, pk); int blockSize = cipher.getBlockSize();// 获得加密块大小,如:加密前数据为128个byte,而key_size=1024 // 加密块大小为127 // byte,加密后为128个byte;因此共有2个加密块,第一个127 // byte第二个为1个byte int outputSize = cipher.getOutputSize(data.length);// 获得加密块加密后块大小 int leavedSize = data.length % blockSize; int blocksSize = leavedSize != 0 ? data.length / blockSize + 1 : data.length / blockSize; byte[] raw = new byte[outputSize * blocksSize]; int i = 0; while (data.length - i * blockSize > 0) { if (data.length - i * blockSize > blockSize) cipher.doFinal(data, i * blockSize, blockSize, raw, i * outputSize); else cipher.doFinal(data, i * blockSize, data.length - i * blockSize, raw, i * outputSize); // 这里面doUpdate方法不可用,查看源代码后发现每次doUpdate后并没有什么实际动作除了把byte[]放到 // ByteArrayOutputStream中,而最后doFinal的时候才将所有的byte[]进行加密,可是到了此时加密块大小很可能已经超出了 // OutputSize所以只好用dofinal方法。 i++; } return raw; } catch (Exception e) { throw new Exception(e.getMessage()); } } /** * * 解密 * * * @param key * 解密的密钥 * * @param raw * 已经加密的数据 * * @return 解密后的明文 * * @throws Exception */ public static byte[] decrypt(PrivateKey pk, byte[] raw) throws Exception { try { Cipher cipher = Cipher.getInstance("RSA", new org.bouncycastle.jce.provider.BouncyCastleProvider()); cipher.init(cipher.DECRYPT_MODE, pk); int blockSize = cipher.getBlockSize(); ByteArrayOutputStream bout = new ByteArrayOutputStream(64); int j = 0; while (raw.length - j * blockSize > 0) { bout.write(cipher.doFinal(raw, j * blockSize, blockSize)); j++; } return bout.toByteArray(); } catch (Exception e) { throw new Exception(e.getMessage()); } } /** * * * * * @param args * * @throws Exception */ public static void main(String[] args) throws Exception { RSAPublicKey rsap = (RSAPublicKey) RSAUtil.generateKeyPair().getPublic(); String test = "hello world"; byte[] en_test = encrypt(getKeyPair().getPublic(),test.getBytes()); byte[] de_test = decrypt(getKeyPair().getPrivate(),en_test); System.out.println(new String(de_test)); } }
2.测试页面:
IndexAction.java
/* * Generated by MyEclipse Struts * Template path: templates/java/JavaClass.vtl */ package com.sunsoft.struts.action; import java.security.interfaces.RSAPrivateKey; import java.security.interfaces.RSAPublicKey; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.struts.action.Action; import org.apache.struts.action.ActionForm; import org.apache.struts.action.ActionForward; import org.apache.struts.action.ActionMapping; import com.sunsoft.struts.util.RSAUtil; /** * MyEclipse Struts * Creation date: 06-28-2008 * * XDoclet definition: * @struts.action validate="true" */ public class IndexAction extends Action { /* * Generated Methods */ /** * Method execute * @param mapping * @param form * @param request * @param response * @return ActionForward */ public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response)throws Exception { RSAPublicKey rsap = (RSAPublicKey) RSAUtil.getKeyPair().getPublic(); String module = rsap.getModulus().toString(16); String empoent = rsap.getPublicExponent().toString(16); System.out.println("module"); System.out.println(module); System.out.println("empoent"); System.out.println(empoent); request.setAttribute("m", module); request.setAttribute("e", empoent); return mapping.findForward("login"); } }
通过此action进入登录页面,并传入公钥的 Modulus 与PublicExponent的hex编码形式。
3。登录页面 login.jsp
<%@ page language="java" pageEncoding="GBK"%> <%@ taglib uri="http://struts.apache.org/tags-bean" prefix="bean" %> <%@ taglib uri="http://struts.apache.org/tags-html" prefix="html" %> <%@ taglib uri="http://struts.apache.org/tags-logic" prefix="logic" %> <%@ taglib uri="http://struts.apache.org/tags-tiles" prefix="tiles" %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html:html lang="true"> <head> <html:base /> <title>login</title> <meta http-equiv="pragma" content="no-cache"> <meta http-equiv="cache-control" content="no-cache"> <meta http-equiv="expires" content="0"> <meta http-equiv="keywords" content="keyword1,keyword2,keyword3"> <meta http-equiv="description" content="This is my page"> <!-- <link rel="stylesheet" type="text/css" href="styles.css"> --> <script type="text/javascript" src="js/RSA.js"></script> <script type="text/javascript" src="js/BigInt.js"></script> <script type="text/javascript" src="js/Barrett.js"></script> <script type="text/javascript"> function rsalogin() { bodyRSA(); var result = encryptedString(key, document.getElementById("pwd").value); //alert(result); loginForm.action="login.do?result="+result; loginForm.submit(); } var key ; function bodyRSA() { setMaxDigits(130); key = new RSAKeyPair("10001","","8c1cd09a04ed01aafe70dc84c5f32ae23a16fe8fc8898aba6797c5a9c708720de4f08dbf086af429fc51c0636208f56de20a8ab5686affd9bdfb643ae1e90d5617155c4867eef06b0884ba8ecd187907c7069ae3eed4f0155eeca6573411864035ae803ad8fd91a0cc479f27e41b19c13465ab30f3cfbfd14de56f49cbd09481"); } </script> </head> <body > <html:form action="login" method="post" focus="username"> <table border="0"> <tr> <td>Login:</td> <td><html:text property="username" /></td> </tr> <tr> <td>Password:</td> <td><html:password property="password" styleId="pwd"/></td> </tr> <tr> <td colspan="2" align="center"><input type="button" value="SUBMIT" onclick="rsalogin();"/></td> </tr> </table> </html:form> </body> </html:html>
3.点击登录后,调用LoginAction.java
/* * Generated by MyEclipse Struts * Template path: templates/java/JavaClass.vtl */ package com.sunsoft.struts.action; import java.math.BigInteger; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.apache.struts.action.Action; import org.apache.struts.action.ActionForm; import org.apache.struts.action.ActionForward; import org.apache.struts.action.ActionMapping; import com.sunsoft.struts.util.RSAUtil; /** * MyEclipse Struts * Creation date: 06-28-2008 * * XDoclet definition: * @struts.action path="/login" name="loginForm" input="/login.jsp" scope="request" validate="true" * @struts.action-forward name="error" path="/error.jsp" * @struts.action-forward name="success" path="/success.jsp" */ public class LoginAction extends Action { /* * Generated Methods */ /** * Method execute * @param mapping * @param form * @param request * @param response * @return ActionForward */ public ActionForward execute(ActionMapping mapping, ActionForm form, HttpServletRequest request, HttpServletResponse response) throws Exception{ //LoginForm loginForm = (LoginForm) form; String result = request.getParameter("result"); System.out.println("原文加密后为:"); System.out.println(result); byte[] en_result = new BigInteger(result, 16).toByteArray(); System.out.println("转成byte[]"+new String(en_result)); byte[] de_result = RSAUtil.decrypt(RSAUtil.getKeyPair().getPrivate(),en_result); System.out.println("还原密文:"); System.out.println(new String(de_result)); StringBuffer sb = new StringBuffer(); sb.append(new String(de_result)); System.out.println(sb.reverse().toString()); return mapping.findForward("success"); } }
因为发现解出的明文是倒序的,后面就用StringBuffer的reverse()来转换了一下。
4。login.jsp所调用的js
- js.rar (6.4 KB)
- 描述: login.jsp所调用的javascript,有: RSA.js BigInt.js Barrett.js
- 下载次数: 2830
评论
28 楼
xuniverse
2016-04-06
hannibal005 写道
楼主,我问下 request.setAttribute("m", module);
request.setAttribute("e", empoent);
这个的作用是把公钥的 Modulus 与PublicExponent传入login.jsp么?我在login.jsp没有看到接受函数,请问JS加密的时候需要这2个参数么?
request.setAttribute("e", empoent);
这个的作用是把公钥的 Modulus 与PublicExponent传入login.jsp么?我在login.jsp没有看到接受函数,请问JS加密的时候需要这2个参数么?
我也想问这个问题,感觉都没用到这两个公钥参数啊
27 楼
kalogen
2015-10-29
gtczr 写道
非常感谢,经过我自己的修改,已经完美实现。
发出来分享一下
26 楼
gtczr
2014-12-16
非常感谢,经过我自己的修改,已经完美实现。
25 楼
baiyun_rain
2014-06-16
[color=cyan][/color][size=small][/size][align=right][/align]
24 楼
lwl_radium
2014-03-06
如大家所说,有些字符js加密后,java解密不了。产生不同的密钥,解密不了的字符串会不同
23 楼
wwilovehome
2013-05-28
楼主,我用JAVA加密,JS解密出现问题,不知道是不是我书写有错。如果方便,希望能够写一个demo发我邮箱:wwilovehome@126.com,谢谢。
22 楼
nokelong
2013-05-21
上面有二错误,bodyRSA中应该是key = new RSAKeyPair("${e}","","${m}");纠结了一天,
一个是encryptedString(key, encodeURIComponent(document.getElementById("pwd").value));用encodeURIComponent转码。大家相互学习
一个是encryptedString(key, encodeURIComponent(document.getElementById("pwd").value));用encodeURIComponent转码。大家相互学习
21 楼
nokelong
2013-05-20
huangshazsw 写道
为什么我后台解密出来的是乱码啊,求解
20 楼
huangshazsw
2013-05-02
为什么我后台解密出来的是乱码啊,求解
19 楼
mingren135
2013-04-19
小心,new 那个provider会出现内存泄露的bug
18 楼
jin8000608172
2013-01-29
楼上的问题我都解决了
17 楼
yangy608
2012-11-15
有个bug,单独加密1或5是解不出来的
16 楼
awerty3450p
2012-05-14
我按六楼的方法 已经解决JAVA解密不正确的问题 但是JAVA加密得到的代码不正确 求破的方法
15 楼
awerty3450p
2012-05-11
博主你好,我有一个字符js加密 和 JAVA加密以后不一致 除了最后一位其他都差1 而另外的字符加密正常,百思不得其解
14 楼
Mov_webhobo
2012-04-19
引用
因为发现解出的明文是倒序的,后面就用StringBuffer的reverse()来转换了一下。
没有看懂JS源码,不知道这个的原因是什么?
引用
3 楼 hdown 2009-07-21 引用
已经在用了,谢谢啦
不过加密个别中英文字符串的时候会抛异常的
真的假的?搞得不敢用了!
13 楼
a47313198
2011-12-26
guzhixiong 写道
楼主,js 加密中文后,解密的是乱码,需要怎么解决呢,,试了好多方法没有解决,请问你有碰到过吗?是怎么解决的,请赐教
先使用 encodeURIComponent() 函数对中文进行编码,再加密试试。
解密后,使用 java.net.Decoder.decode(str, "UTF-8") 进行中文还原。
12 楼
hannibal005
2011-11-06
还是觉得JS里面的加密方法有问题。。因为我java里面的源程序加密解密字符串13579都没问题
11 楼
hannibal005
2011-11-05
这个方法很多字符都不能正常解密,比如13579 等等
10 楼
hannibal005
2011-11-05
楼主,我问下 request.setAttribute("m", module);
request.setAttribute("e", empoent);
这个的作用是把公钥的 Modulus 与PublicExponent传入login.jsp么?我在login.jsp没有看到接受函数,请问JS加密的时候需要这2个参数么?
request.setAttribute("e", empoent);
这个的作用是把公钥的 Modulus 与PublicExponent传入login.jsp么?我在login.jsp没有看到接受函数,请问JS加密的时候需要这2个参数么?
9 楼
guzhixiong
2011-09-07
楼主,js 加密中文后,解密的是乱码,需要怎么解决呢,,试了好多方法没有解决,请问你有碰到过吗?是怎么解决的,请赐教
发表评论
-
高级Java程序员值得拥有的10本书
2015-05-04 07:24 766Java是时下最流行的编程语言之一。市面上也出现了适合初学者 ... -
深入理解java异常处理机制
2015-01-30 09:30 12801. 引子 try…catch…fi ... -
java 运行时参数设置
2015-01-07 09:13 823JVM的运行时参数: -Xms为执行单元内存的1/4, ... -
每个Java开发者都应该知道的5个JDK工具
2014-12-29 12:37 1103JDK是Java语言的软件开 ... -
使用双重锁判定可以大幅降低锁的征用
2014-12-29 12:30 707class ObjInstance { //单例 pri ... -
MAVEN Scope使用说明
2014-11-24 09:40 716在Maven的依赖管理中,经常会用到依赖的scope设置。这 ... -
Spring4 quartz job xml configuration
2014-11-11 09:46 14051. 定义job details public ... -
Add items into list in one line using guava
2014-11-10 10:54 694//@formatter:off fina ... -
配置动态读取(变化)文件 in Spring
2014-11-10 08:51 12931. 从环境变量中读取路径: <bean id=&q ... -
JAVA实现AES加密与解密
2014-11-04 15:34 616package com.eifesun.monitor.up ... -
Netty4.x分析
2014-07-31 11:06 1421官网定义: netty是一个异步、事件驱动的网络应用框架,用 ... -
Ways to sort lists of objects in Java based on multiple fields
2014-07-21 17:19 7351. the first way: Sorting wit ... -
how to parse a String to BigDecimal
2014-07-21 10:08 889private BigDecimal parsePrice( ... -
order list using google guava
2014-07-21 09:08 854Predicate<String> filter ... -
Java 读文件操作
2014-07-08 14:09 8581. only use java core, no exte ... -
怎样使Java 中测试按一定顺序执行
2014-03-10 11:27 1268@FixMethodOrder(MethodSorters. ... -
如何实现在当类初始化时,自动调动某个方法
2014-02-14 14:44 925有两种思路, 1. 将这个类实现为thread类 (or ... -
持续集成JenkinsAPI常见用法
2014-02-10 13:54 43jenkins(持续集成开源工具)提供了丰富的api接口,基 ... -
Sonar 安装与使用
2014-01-13 10:49 1695Sonar 是一个用于代码质量管理的开放平台。通过插件机制, ... -
源代码管理分析工具 Source Navigator的安装与使用
2014-01-13 09:51 1843Source-Navigator是原来redhat开发的一个 ...
相关推荐
javascript java rsa 加密 解密 数据加密 数据解密
RSA 使用java 和javascript进行加解密
综合网上javasript和java RSA加密解密资源,实现了java和javascript RSA加密解密的互操作,都可以生成公钥传给对方加密然后回传密文用自己的私钥解密。
使用RSA非对称加密完成JavaScript前端RSA加密和分段加解密,最近研究了RSA非对称加密,关于什么是RSA,网上各种文章一搜一大把,但是关于如何使用RSA完成前端的组合加密解密,东西就非常少了,并且由于RSA的特性,一...
RSA通过javascript加密java解密
这是我有自己的项目中使用的加密解密代码,下载后按说明放到自己的项目中就能使用,如有问题请留言给我,我会第一时间回复大家。
RSA加密解密
RSA非对称 C#解密、js加密实现登陆密文,传输RSA非对称 C#解密、js加密实现登陆密文传输,RSA非对称 C#解密、js加密实现登陆密文传输
注意:AES用的是CryptoJS库的,而RSA则是用的jsencrypt库,RSA用的公钥和私钥是PEM格式的,其他格式无法通过,同时只能支持公钥加密,私钥解密。tips:支持跨语言,支持js与.net后台的加密以及解密,如需借鉴后台...
jsencrypt.min.js通过JSEncrypt分段加密解密,自己添加了分段加解密的方法
//加密 var encrypt = new JSEncrypt(); encrypt.setPublicKey(PUBLIC_KEY); var encrypted = encrypt.encryptLong("results"); console.log('加密后数据:%o', encrypted); //解密 var decrypt = new JSEncrypt(); ...
本rsa算法是使用Java与javascript加密解密范例代码,该资料从互联网收集,加上了自己的使用体会,如果对你有帮助那是万幸! js加密部分
本rsa算法是使用Java与javascript加密解密范例代码,该资料从互联网收集,加上了自己的使用体会,如果对你有帮助那是万幸!
运用网页设计的方式写了一个RSA加密小工具 有浏览器能看网页的机器上都能用
在客户端浏览器,Javascript使用RSA算法,以公钥对密码进行加密,服务端使用相应的私钥进行解密。一般用于注册时或登录时填写的密码。 Java引用到的包: commons-lang bouncycastle slf4j commons-codec ...
参考:Blackberry10 使用js+HTML5开发 RSA加密解密时遇到的问题:BB10端使用js加密与解密, 服务器端使用加密与解密 1,加密非常的简单代码机会上没怎么修改,另外js加密可能出现的问题在BB10 AES加密中已经说过,js ...
通过javascript实现RSA加密,使用了标准的国际PKCS1填充,支持UTF-8编码