当前位置: 首页 > 图文教程 > Java技术 > Java基础 > Java加密和数字签名编程(2)

Java基础
给Java开发者的10条戒律
Java中调用C/C++生成的DLL
Java事务处理类型/差异总结
使用DataAdapter对象
基于Java事务模型的事务策略
Java的多进程运行模式分析
JAVA反射机制
垃圾清理势在必行:java垃圾收集算法
Clone 方法的使用(深度克隆)
dwr-spring-hibernate的使用
java Spring学习记录
Java性能的十一个用法
java中的assert关键字
Java中的一直不能掌握的switch-case语句
kernighthen-lin算法:我自己编的
Hashtable和HashMap的区别
一个Java程序员应该掌握的10项技能
通过线程门实现Java并发编程
java中的assert关键字
细节的积累 ---字符串的常用方法总结

Java基础 中的 Java加密和数字签名编程(2)


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2010-01-23   浏览: 111 ::
收藏到网摘: n/a

上篇文章说到,私钥加密与公钥加密,下面继续和大家分享下Java加密和数字签名编程,希望对大家有所帮助。

4)数字签名:

 

  数字签名,它是确定交换消息的通信方身份的第一个级别。上面A通过使用公钥加密数据后发给BB利用私钥解密就得到了需要的数据,问题来了,由于都是使用公钥加密,那么如何检验是A发过来的消息呢?上面也提到了一点,私钥是唯一的,那么A就可以利用A自己的私钥进行加密,然后B再利用A的公钥来解密,就可以了;数字签名的原理就基于此,而通常为了证明发送数据的真实性,通过利用消息摘要获得简短的消息内容,然后再利用私钥进行加密散列数据和消息一起发送。java中为数字签名提供了良好的支持,java.security.Signature类提供了消息签名:

 

import java.security.Signature;

import java.security.KeyPairGenerator;

import java.security.KeyPair;

import java.security.SignatureException;

 

/**

*数字签名,使用RSA私钥对对消息摘要签名,然后使用公鈅验证 测试

*/

public class DigitalSignature2Example{

 public static void main(String[] args) throws Exception{

  if(args.length!=1){

   System.err.println("Usage:java DigitalSignature2Example ");

   System.exit(1);

  }

 

  byte[] plainText=args[0].getBytes("UTF8");

  //形成RSA公钥对

  System.out.println("nStart generating RSA key");

  KeyPairGenerator keyGen=KeyPairGenerator.getInstance("RSA");

  keyGen.initialize(1024);

 

  KeyPair key=keyGen.generateKeyPair();

  System.out.println("Finish generating RSA key");

  //使用私鈅签名

  Signature sig=Signature.getInstance("SHA1WithRSA");

  sig.initSign(key.getPrivate());

  sig.update(plainText);

  byte[] signature=sig.sign();

  System.out.println(sig.getProvider().getInfo());

  System.out.println("nSignature:");

  System.out.println(new String(signature,"UTF8"));

 

  //使用公鈅验证

  System.out.println("nStart signature verification");

  sig.initVerify(key.getPublic());

  sig.update(plainText);

  try{

   if(sig.verify(signature)){

    System.out.println("Signature verified");

   }else System.out.println("Signature failed");

   }catch(SignatureException e){

    System.out.println("Signature failed");

   }

  }

}

  5)数字证书。

 

  还有个问题,就是公钥问题,A用私钥加密了,那么B接受到消息后,用A提供的公钥解密;那么现在有个讨厌的C,他把消息拦截了,然后用自己的私钥加密,同时把自己的公钥发给B,并告诉B,那是A的公钥,结果....,这时候就需要一个中间机构出来说话了(相信权威,我是正确的),就出现了Certificate Authority(也即CA),有名的CA机构有Verisign等,目前数字认证的工业标准是:CCITTX.509

数字证书:它将一个身份标识连同公钥一起进行封装,并由称为认证中心或 CA 的第三方进行数字签名。

 

  密钥库:java平台为你提供了密钥库,用作密钥和证书的资源库。从物理上讲,密钥库是缺省名称为 .keystore 的文件(有一个选项使它成为加密文件)。密钥和证书可以拥有名称(称为别名),每个别名都由唯一的密码保护。密钥库本身也受密码保护;您可以选择让每个别名密码与主密钥库密码匹配。

 

  使用工具keytool,我们来做一件自我认证的事情吧(相信我的认证):

 

  1、创建密钥库keytool -genkey -v -alias feiUserKey -keyalg RSA 默认在自己的home目录下(windows系统是c:documents and settings<你的用户名> 目录下的.keystore文件),创建我们用 RSA 算法生成别名为 feiUserKey 的自签名的证书,如果使用了-keystore mm 就在当前目录下创建一个密钥库mm文件来保存密钥和证书。

 

  2、查看证书:keytool -list 列举了密钥库的所有的证书

 

  也可以在dos下输入keytool -help查看帮助。

  二、JAR的签名

 

  我们已经学会了怎样创建自己的证书了,现在可以开始了解怎样对JAR文件签名,JAR文件在Java中相当于 ZIP 文件,允许将多个 Java 类文件打包到一个具有 .jar 扩展名的文件中,然后可以对这个jar文件进行数字签名,以证实其来源和真实性。该 JAR 文件的接收方可以根据发送方的签名决定是否信任该代码,并可以确信该内容在接收之前没有被篡改过。同时在部署中,可以通过在策略文件中放置访问控制语句根据签名者的身份分配对机器资源的访问权。这样,有些Applet的安全检验访问就得以进行。

 

  使用jarsigner工具可以对jar文件进行签名:

 

  现在假设我们有个Test.jar文件(可以使用jar命令行工具生成):

 

  jarsigner Test.jar feiUserKey (这里我们上面创建了该别名的证书) ,详细信息可以输入jarsigner查看帮助

 

  验证其真实性:jarsigner -verify Test.jar(注意,验证的是jar是否被修改了,但不检验减少的,如果增加了新的内容,也提示,但减少的不会提示。)

 

  使用Applet中::<applet code="Test.class" archive="Test.jar" width="150" height="100"></applet>然后浏览器就会提示你:准许这个会话-拒绝-始终准许-查看证书等。

 

  三、安全套接字层(SSL Secure Sockets Layer)和传输层安全性(TLS Transport Layer Security

 

  安全套接字层和传输层安全性是用于在客户机和服务器之间构建安全的通信通道的协议。它也用来为客户机认证服务器,以及(不太常用的)为服务器认证客户机。该协议在浏览器应用程序中比较常见,浏览器窗口底部的锁表明 SSL/TLS 有效:

 

  1)当使用 SSL/TLS(通常使用 https:// URL)向站点进行请求时,从服务器向客户机发送一个证书。客户机使用已安装的公共 CA 证书通过这个证书验证服务器的身份,然后检查 IP 名称(机器名)与客户机连接的机器是否匹配。

 

  2)客户机生成一些可以用来生成对话的私钥(称为会话密钥)的随机信息,然后用服务器的公钥对它加密并将它发送到服务器。服务器用自己的私钥解密消息,然后用该随机信息派生出和客户机一样的私有会话密钥。通常在这个阶段使用 RSA 公钥算法。

 

  3)客户机和服务器使用私有会话密钥和私钥算法(通常是 RC4)进行通信。使用另一个密钥的消息认证码来确保消息的完整性。

 

  javajavax.net.ssl.SSLServerSocketFactory类提供了一个很好的SSLServerSocker的工厂类,熟悉Socket编程的读者可以去练习。当编写完服务器端之后,在浏览器上输入https://主机名:端口 就会通过SSL/TLS进行通话了。注意:运行服务端的时候要带系统环境变量运行:javax.net.ssl.keyStore=密钥库(创建证书时,名字应该为主机名,比如localhost)javax.net.ssl.keyStorePassword=你的密码