Java实验报告(实验四)

北京电子科技学院(BESTI)

                        实     验    报     告

课程:Java    班级:1352班      姓名:王国伊    学号:20135207

成绩:             指导教师:娄嘉鹏       实验日期:2015.6.9

实验密级:无       预习程度:           实验时间:15:30-18:00

仪器组次:         必修/选修:选修     实验序号:四

实验名称:  服务器与客户端间传送信息加解密                                                   

实验目的与要求: 1.没有Linux基础的同学建议先学习《Linux基础入门(新版)》《Vim编辑器》 课程。

2.使用实验楼git服务托管代码。

3.完成实验、撰写实验报告,实验报告以博客方式发表在博客园,注意实验报告重点是运行结果,遇到的问题(工具查找,安装,使用,程序的编辑,调试,运行等)、解决办法(空洞的方法如“查网络”、“问同学”、“看书”等一律得0分)以及分析(从中可以得到什么启示,有什么收获,教训等)。报告可以参考范飞龙老师的指导

4. 严禁抄袭,有该行为者实验成绩归零,并附加其他惩罚措施。

实验仪器:

名称

型号

数量

笔记本电脑

Lenovo Z485

1台

 

 

 

 

实验内容、步骤与体会(附纸):

一、实验内容

1. 先运行教材上TCP代码,一人服务器,一人客户端。

2. 下载加解密代码,先编译运行代码,一人加密一人解密,适当修改代码。

3. 然后集成代码,一人加密后通过TCP发送,加密使用AES或DES,AES或DES加密密钥Key的发送,使用服务器的公钥加密,公钥算法用RSA或DH,发送信息的完整性验证使用MD5或SHA3。

4.使用试验后git服务托管代码。

5.完成实验后,找老师验收,课下写Blog。

 

二、实验步骤

结对同学:20135115 臧文君

Blog网址:http://www.cnblogs.com/CatherineZang/

1、实验分工:臧文君---服务器,加解密代码整合。

                  王国伊---客户端,实现两台PC机相连。

2、实验步骤:

(1)下载老师提供的加解密代码,分析理解代码中语句的功能。

(2)根据代码中的语句,查找自己主机的IP地址,确定端口号。

     在命令行中输入:ipconfig。

 

找到自己主机的IP地址为:222.28.128.119。

(3)根据书上TCP的实例,用BufferedReader获取从服务器传入的数据流,再用PrintWriter获取从客户端传出的输出流。

(4)用RSA算法加密,加密使用服务器的公钥,再将加密后的密钥传给服务器。

(5)用DES算法加密输入的明文,将加密后的密文传给服务器。

(6)使用Hash函数,计算明文的Hash函数值,传给服务器。

(7)把从服务器返回的结果输出。

 

 

3、代码:

服务器:

package exp4;

 

import java.net.*;

import java.io.*;

import java.security.*;

import java.security.spec.*;

import javax.crypto.*;

import javax.crypto.spec.*;

import javax.crypto.interfaces.*;

import java.security.interfaces.*;

import java.math.*;

 

public class Server {

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

       ServerSocket sc = null;

       Socket socket = null;

       try {

           sc = new ServerSocket(8029);// 创建服务器套接字

           System.out.println("端口号:" + sc.getLocalPort());

           System.out.println("服务器已经启动...");

           socket = sc.accept(); // 等待客户端连接

           System.out.println("已经建立连接");

           // 获得网络输入流对象的引用

           BufferedReader in = new BufferedReader(new InputStreamReader(

                  socket.getInputStream()));

           // //获得网络输出流对象的引用

           PrintWriter out = new PrintWriter(new BufferedWriter(

                  new OutputStreamWriter(socket.getOutputStream())), true);

 

           String aline2 = in.readLine();

           BigInteger c = new BigInteger(aline2);

           FileInputStream f = new FileInputStream("Skey_RSA_priv.dat");

           ObjectInputStream b = new ObjectInputStream(f);

           RSAPrivateKey prk = (RSAPrivateKey) b.readObject();

           BigInteger d = prk.getPrivateExponent();

           BigInteger n = prk.getModulus();

           BigInteger m = c.modPow(d, n);

           byte[] keykb = m.toByteArray();

           String aline = in.readLine();// 读取客户端传送来的数据

           byte[] ctext = parseHexStr2Byte(aline);

           Key k = new SecretKeySpec(keykb, "DESede");

           Cipher cp = Cipher.getInstance("DESede");

           cp.init(Cipher.DECRYPT_MODE, k);

           byte[] ptext = cp.doFinal(ctext);

 

           String p = new String(ptext, "UTF8");

           System.out.println("从客户端接收到信息为:" + p); // 通过网络输出流返回结果给客户端

 

           String aline3 = in.readLine();

           String x = p;

           MessageDigest m2 = MessageDigest.getInstance("MD5");

           m2.update(x.getBytes());

           byte a[] = m2.digest();

           String result = "";

           for (int i = 0; i < a.length; i++) {

              result += Integer.toHexString((0x000000ff & a[i]) | 0xffffff00)

                     .substring(6);

           }

           System.out.println(result);

 

           if (aline3.equals(result)) {

              System.out.println("匹配成功");

           }

 

           out.println("匹配成功");

           out.close();

           in.close();

           sc.close();

       } catch (Exception e) {

           System.out.println(e);

       }

    }

 

    public static String parseByte2HexStr(byte buf[]) {

       StringBuffer sb = new StringBuffer();

       for (int i = 0; i < buf.length; i++) {

           String hex = Integer.toHexString(buf[i] & 0xFF);

           if (hex.length() == 1) {

              hex = '0' + hex;

           }

           sb.append(hex.toUpperCase());

       }

       return sb.toString();

    }

 

    public static byte[] parseHexStr2Byte(String hexStr) {

       if (hexStr.length() < 1)

           return null;

       byte[] result = new byte[hexStr.length() / 2];

       for (int i = 0; i < hexStr.length() / 2; i++) {

           int high = Integer.parseInt(hexStr.substring(i * 2, i * 2 + 1), 16);

           int low = Integer.parseInt(hexStr.substring(i * 2 + 1, i * 2 + 2),

                  16);

           result[i] = (byte) (high * 16 + low);

       }

       return result;

    }

}

客户端:

package exp4;

 

import java.net.*;

import java.io.*;

import java.security.*;

import javax.crypto.*;

import javax.crypto.spec.*;

import java.security.spec.*;

import javax.crypto.interfaces.*;

import java.security.interfaces.*;

import java.math.*;

 

public class Client {

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

       try {

           KeyGenerator kg = KeyGenerator.getInstance("DESede");

           kg.init(168);

           SecretKey k = kg.generateKey();

           byte[] ptext2 = k.getEncoded();

 

           // 创建连接特定服务器的指定端口的Socket对象

           Socket socket = new Socket("222.28.128.119", 8028);

           // 网络输入流

           BufferedReader in = new BufferedReader(new InputStreamReader(

                  socket.getInputStream()));

           // 网络输出流

           PrintWriter out = new PrintWriter(new BufferedWriter(

                  new OutputStreamWriter(socket.getOutputStream())), true);

           // 创建键盘输入流

           BufferedReader stdin = new BufferedReader(new InputStreamReader(

                  System.in));

 

           FileInputStream f3 = new FileInputStream("Skey_RSA_pub.dat");

           ObjectInputStream b2 = new ObjectInputStream(f3);

           RSAPublicKey pbk = (RSAPublicKey) b2.readObject();

           BigInteger e = pbk.getPublicExponent();

           BigInteger n = pbk.getModulus();

           BigInteger m = new BigInteger(ptext2);

           BigInteger c = m.modPow(e, n);

           String cs = c.toString();

           out.println(cs); // 通过网络传送到服务器

 

           System.out.print("请输入待发送的数据:");

           String s = stdin.readLine();

           Cipher cp = Cipher.getInstance("DESede");

           cp.init(Cipher.ENCRYPT_MODE, k);

           byte ptext[] = s.getBytes("UTF8");

           byte ctext[] = cp.doFinal(ptext);

           String str = parseByte2HexStr(ctext);

           out.println(str);

 

           String x = s;

           MessageDigest m2 = MessageDigest.getInstance("MD5");

           m2.update(x.getBytes());

           byte a[] = m2.digest();

           String result = "";

           for (int i = 0; i < a.length; i++) {

              result += Integer.toHexString((0x000000ff & a[i]) | 0xffffff00)

                     .substring(6);

           }

           System.out.println(result);

           out.println(result);

 

           str = in.readLine();// 从网络输入流读取结果

           System.out.println("从服务器接收到的结果为:" + str); // 输出服务器返回的结果

       } catch (Exception e) {

           System.out.println(e);

       } finally {

       }

 

    }

 

    public static String parseByte2HexStr(byte buf[]) {

       StringBuffer sb = new StringBuffer();

       for (int i = 0; i < buf.length; i++) {

           String hex = Integer.toHexString(buf[i] & 0xFF);

           if (hex.length() == 1) {

              hex = '0' + hex;

           }

           sb.append(hex.toUpperCase());

       }

       return sb.toString();

    }

 

    public static byte[] parseHexStr2Byte(String hexStr) {

       if (hexStr.length() < 1)

           return null;

       byte[] result = new byte[hexStr.length() / 2];

       for (int i = 0; i < hexStr.length() / 2; i++) {

           int high = Integer.parseInt(hexStr.substring(i * 2, i * 2 + 1), 16);

           int low = Integer.parseInt(hexStr.substring(i * 2 + 1, i * 2 + 2),

                  16);

           result[i] = (byte) (high * 16 + low);

       }

       return result;

    }

}

 

 

三、实验体会

本次实验是java课的最后一次实验,首先十分感谢娄老师对我们的殷切指导与教育。虽然过程中有一些觉得困难的时候。但最后的确学到一些很实用的技巧与编程思想。这使得对理解计算机语言思想有很大帮助。特别是对于面向对象的思想对于无论是理解现实生活的问题还是计算机需要处理的问题都是一种很有效的思想。

对于本次实验,相比于上次的XP实践来讲有提升一个层次,要求在网络中实现。而且传输数据要求加密,这与我们的专业有息息相关,无论是加解密代码还是客户端服务器的建立,都有很大的实际应用意义。在与伙伴的网络连接中,总会出现无效长度的情况。后来将RSA的全部dat文件放到程序调用路径中又更换了端口重新连接几次了才可以传输数据。最终完成了实验。

 

 

附:

步骤

耗时

百分比

需求分析

0.5h

5%

设计

1.5h

15%

代码实现

5h

50%

测试

2h

20%

分析总结

1h

10%

转载于:https://www.cnblogs.com/20135207oneking/p/4572670.html

实验1 Java语言基础 4 一、实验目的 4 二、实验要求 4 三、实验内容 4 (一) 声明不同数据类型的变量 4 (二) 了解变量的使用范围 5 (三) 使用关系运算符和逻辑运算符 5 () 使用表达式语句与复合语句 6 (五) 使用选择语句 6 (六) 使用循环语句 8 实验2 面向对象编程 11 一、实验目的 11 二、实验要求 11 三、实验内容 11 (一)创建 Applet 应用小程序 11 (二)创建对象并使用对象 12 (三)编写显示当前日期和时间的程序 13 ()使用修饰符 14 (五)方法中参数传递 15 (六)类的继承性 16 (七)类的多态性 18 实验3 包、接口与异常处理 22 一、实验目的 22 二、实验要求 22 三、实验内容 22 (一)了解并使用 Java 的系统包 22 (二)创建并使用自定义包 22 (三)使用接口技术 24 ()了解异常处理机制 25 实验4 常用系统类的使用 27 一、实验目的 27 二、实验要求 27 三、实验内容 27 (一)了解 Applet 的生命周期 27 (二)使用数学函数类 28 (三)使用日期类 29 实验5 建立图形用户界面 32 一、实验目的 32 二、实验要求 32 三、实验内容 32 (一)创建图形用户界面 32 (二)了解事件处理机制 34 (三)建立独立运行的窗口界面并使用匿名类 36 ()使用 Swing 组件 39 (五)使用自定义对话框与内部类 41 实验6 图形处理 44 一、实验目的 44 二、实验要求 44 三、实验内容 44 (一)使用图形类 Graphics 44 实验7图形处理(二) 47 一、实验目的 47 二、实验要求 47 三、实验内容 47 (一)幻灯机效果——连续显示多幅图像 47 (二)使用滚动条改变背景颜色 48 (三)Applet 与 Application 合并运行 49 ()创建电闪雷鸣的动画 50 实验8 流与文件 53 一、实验目的 53 二、实验要求 53 三、实验内容 53 (一)使用标准数据流的应用程序 53 (二)使用文件输入输出流的应用程序 54 (三)使用随机文件类的应用程序 54 ()使用数据输入输出流与文件输入输出流类的应用程序 55 (五)使用对象输入输出流的应用程序 56 实验9 线程 58 一、实验目的 58 二、实验要求 58 三、实验内容 58 (一)Thread子类的方法实现多线程 58 (二)实现Runnable接口的方法实现多线程 59 实验10 数据库的连接:JDBC 61 一、实验目的 61 二、实验要求 61 三、实验内容 61 (一)配置ODBC数据源 61 (二)编写程序,按照下表的结构建立"student"表 61 (三)编写程序,完成填写功能 62 ()编写程序,完成查询功能 62
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值