20232402 2023-2024-2 《Python程序设计》实验三报告

20232402 2023-2024-2 《Python程序设计》实验三报告

课程:《Python程序设计》
班级: 2324
姓名: 王艺瑶
学号:20232402
实验教师:王志强
实验日期:2024年4月17日
必修/选修: 公选课

1.实验内容

创建服务端和客户端,服务端在特定端口监听多个客户请求。客户端和服务端通过Socket套接字(TCP/UDP)进行通信。

2. 实验过程及结果

(1)概念理解
  • 套接字(Socket):一个网络通信的端点,提供了在计算机网络中进行双向数据传输的机制。它能够实现不同计算机之间或同一计算机内不同进程之间的通信。
    在网络编程中,套接字用于连接网络上的两个程序,一个程序在某台计算机上运行并作为客户端,另一个程序在另一台计算机上运行并作为服务器。通过网络协议(通常是TCP/IP)建立连接后,两个程序就可以通过套接字交换数据。
  • TCP服务端:或称为TCP服务器,是在基于TCP(Transmission Control Protocol,传输控制协议)的网络通信中,作为服务提供方的计算机或程序。它通过TCP协议监听来自客户端的连接请求,建立连接后进行数据的接收和发送,以此提供网络服务。
    在整个过程中,TCP服务端需要处理各种网络事件,如连接的建立和释放、数据的读取和发送等。对于支持多客户端连接的服务端而言,通常会采用多线程或多进程的方式来同时处理多个客户端的请求,确保服务的响应性和可用性。
  • DES:全称为Data Encryption Standard,是一种以64位为分组长度,密钥长度为56位(实际使用64位,但有8位用于奇偶校验)的对称密钥加密算法。使用DES加密和解密操作,使用的密钥必须是8字节(即64位)长,CBC模式要求有一个初始向量(IV),在该实验中,IV和密钥是一样的,但在实际应用中,IV通常应该是随机的,以增加安全性。PKCS5是一种常用的填充模式,用于在需要时填充数据块到适合的大小。
(2)代码编写
  • 编写DES加密和解密操作文件:DesModule.py :
from pyDes import *  # *表示引用所有函数
import binascii

# DES 加密函数
def des_encrypt(s, KEY):
    secret_key = KEY  # 设置密钥
    iv = secret_key   # 使用密钥作为初始化向量
    k = des(secret_key, CBC, iv, pad=None, padmode=PAD_PKCS5)  # 创建DES加密对象,CBC模式,使用PKCS5填充
    en = k.encrypt(s, padmode=PAD_PKCS5)  # 加密数据
    return binascii.b2a_hex(en).decode()  # 将加密后的二进制数据转换为十六进制字符串并返回

# DES 解密函数
def des_descrypt(s, KEY):
    secret_key = KEY  # 设置密钥
    iv = secret_key   # 使用密钥作为初始化向量
    k = des(secret_key, CBC, iv, pad=None, padmode=PAD_PKCS5)  # 创建DES解密对象,CBC模式,使用PKCS5填充
    de = k.decrypt(binascii.a2b_hex(s), padmode=PAD_PKCS5)  # 将十六进制字符串转换为二进制后解密
    return de.decode()  # 解密后返回字符串形式的明文

# 测试代码
if __name__ == "__main__":
    ciphertext = des_encrypt("hello, world!", "12345678")  # 对明文"hello, world!"使用密钥"12345678"进行加密
    print("密文:", ciphertext)  # 打印加密后的密文
    plaintext = des_descrypt(ciphertext, "12345678")  # 使用相同的密钥对密文进行解密
    print("解密后的明文:", plaintext)  # 打印解密后的明文`
  • 编写单次通信服务端文件:SocketServer.py:
import socket
from DesModule import des_descrypt, des_encrypt

# 创建一个 socket 对象用于网络通信,AF_INET 指使用 IPv4 地址,SOCK_STREAM 指使用 TCP 连接
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.bind(("172.16.211.247", 8000))  # 绑定 IP 地址和端口号
s.listen()  # 监听客户端的连接请求
# accept 方法等待客户端的连接,这是一个阻塞调用,返回连接的 socket 对象和对方的地址信息
conn, address = s.accept()
data = conn.recv(1024)  # 接收,最大接收量为 1024 字节
plaintext = des_descrypt(data.decode(), "12345678")  # 解码,解密
print("从客户端收到的信息为:", plaintext)
ciphertext = des_encrypt("20232402", "12345678")  # 加密
conn.sendall(ciphertext.encode())  # 编码,发送

s.close()  # 关闭 socket 连接
  • 编写单次通信客户端文件:SocketClient.py:
import socket
import os
from DesModule import *

file1 = open("秘密", "w+")
ciphertext = des_encrypt("hello~", "12345678")
file1.write(ciphertext)
file1.seek(0)  # 指针回到文件的开始位置,以便从头读取文件
cryptostring = file1.readline()  # 读取文件的第一行,此处应该是加密后的字符串
print(cryptostring)

s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.connect(("172.16.208.90", 2025))
s.sendall(cryptostring.encode())  # 编码,发送
data = s.recv(1024)  # 接收,且最多接收1024字节
plaintext = des_descrypt(data.decode(), "12345678")
print("从服务器接收的数据为:", plaintext)

s.close()
  • 编写多次通信服务端文件:SocketServer4ever.py:
import socket
from DesModule import des_descrypt, des_encrypt

# 创建一个TCP socket对象
tcpserver = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcpserver.bind(("172.16.211.247", 9000))
tcpserver.listen(5)  # 监听,最多可以有5个客户端等待连接

while True:  # 无限循环,使服务器能够持续接收客户端的连接
    conn, addr = tcpserver.accept()
    print(conn)  # 打印连接对象
   
    while True:   # 无限循环,以持续从连接的客户端接收数据
        try:
            data = conn.recv(1024)
            plaintext = des_descrypt(data.decode("utf-8"), "12345678")
            print(plaintext)
            ret_data = input("请回复:")
            ciphertext = des_encrypt(ret_data, "12345678")
            conn.send(ciphertext.encode('utf-8'))
        except Exception:   # 如果在数据传输过程中出现异常,则跳出这次循环,等待下一个连接
            break
    
    conn.close()
  • 编写多次通信用户端文件:SocketClient-4ever.py
import socket
from DesModule import des_descrypt,des_encrypt
tcpclient = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcpclient.connect(("172.16.208.90", 2025))
while True:
    while True:
        data = input('>>>').strip()  # 显示提示符 >>> 等待用户输入并去除首尾的空白字符。
        ciphertext = des_encrypt(data, "12345678")
        tcpclient.send(ciphertext.encode('utf-8'))
        ret = tcpclient.recv(1024)
        plaintext = des_descrypt(ret.decode('utf-8'), "12345678")
        print(plaintext)
tcpclient.close()
(3)代码运行
  • 测试des加解密算法代码:
    在这里插入图片描述

  • 作服务端执行单次通信:请添加图片描述

  • 作客户端执行单次通信:请添加图片描述

  • 作服务端执行多次通信:请添加图片描述

  • 作客户端执行多次通信:
    请添加图片描述

(4)将代码上传至gitee

在这里插入图片描述

在这里插入图片描述

3. 实验过程中遇到的问题和解决过程

  • 问题1:加解密时报错
  • 问题1解决方案:发送数据时,应当先加密后编码;而接收数据时,应当先解码后解密。因为加解密算法需要接收字符串,而通信需要接收utf-8编码后的数据。
  • 问题2:网络无法连接
  • 问题2解决方案:首先应当windows+R进入cmd,输入igconfig查询无线局域网适配器 WLAN的IPv4 地址。并且保证服务器端输入自己连的WLAN的IPv4 地址,客户端与服务器端地址保持一致,同时服务器端也要连上同样的WLAN。在这里插入图片描述

4. 其他(感悟、思考等)

1.python真是太有意思了,可以用很简单的语句实现很多有趣的功能。
2.上学期学网安导论的时候在ubuntu中做过sm加解密的实验,感觉对加解密还是一知半解,这次实验之后茅塞顿开。

5. 参考资料

gitee中的代码

  • 17
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值