客户端向服务器发送一个请求,请求内容是一个文件名,服务器在查找自己这边有没有这个文件,如果有的话就发送给客户端

1、客户端

  1. 生成socket对象
  2. 建立连接
  3. 输入想要接收的文件
  4. 将输入的文件名发送给服务器
  5. 接收服务器发回的关于即将要发送来的文件的大小
  6. 发送一条信息给服务器告诉它准备好接收了
  7. 接收文件数据
  8. 打印全部接收的提示信息

client具体实现的代码如下:

# Author: Mr.Xue
# 2019.10.29
# socket_ftp_client.py

import socket
import hashlib

client = socket.socket() # 生成socket连接对象
client.connect(('localhost', 6961)) # 建立连接

while True:
 cmd = input("").strip() # 输入想要接收的文件
 if len(cmd) == 0: continue # 输入为空,重新再输
 if cmd.startswith('get'): # 判断指令是否以get开头
 client.send(cmd.encode("utf-8")) # 发送
 server_response = client.recv(1024) #接收即将发送来的文件的大小
 print("server response:", server_response) # 打印文件的大小
 client.send(b'ready to recv file...') # 发送消息告诉服务已经准备好接收了
 file_total_size = int(server_response.decode()) # 记录文件的总大小
 received_size = 0 # 记录已经接收了的文件的大小
 filename = cmd.split()[1] # 取出输入的文件名
 f = open(filename + '.new', 'wb') # 新建一个本地文件来存储接收的数据
 m = hashlib.md5() # md5加密
 while received_size < file_total_size:
  data = client.recv(1024) # 接收数据,一次最大接收1024bytes
  received_size += len(data) # 记录已接收的数据大小
  m.update(data) # 用md5加密
  f.write(data) # 写入文件
 else:
  new_file_md5 = m.hexdigest() # 16进制显示加密文件
  print("file recv done", received_size, file_total_size)
  f.close()
  server_md5 = client.recv(1024) # 接收服务器端以md5加密的这个接收文件的加密文件,拿来和接收之后的加密文件做比较
  print("md5", new_file_md5, server_md5)
client.close() 

client端比较难的一点和上篇一样,还是如何判断服务器端要发送的数据,客户端是否完全都接收过来了,实现逻辑在上面代码中,也可参考前一篇socket(二);在客户端这一次多了一个新的知识点,用md5来加密接收到的文件,先生成md5实例对象,再调用update()来加密文件,最后和服务器端的加密文件进行比较,看看是不是一样的。

2、服务器

  1. 生成socket连接对象
  2. 绑定要监听端口
  3. 监听
  4. 等待客户端的连接
  5. 接收客户端发来的文件名
  6. 查找这个文件是否存在
  7. 如果存在,打开文件计算大小,发送给客户端
  8. 等待客户端确认
  9. 发送文件具体内容给客户端

server具体实现代码如下:

# Author: Mr.Xue
# 2019.10.29
# socket_ftp_server.py

import socket, os, hashlib

server = socket.socket() #生成socket对象
server.bind(('localhost', 6961)) # 绑定ip和端口
server.listen() #监听
while True:
 conn, addr = server.accept() # 等待客户端连接
 while True:
 print("等待接收文件名...")
 data = conn.recv(1024) # 接收客户端发来的信息
 if not data: #判断客户端是否断连
  print('lost a link...')
  break
 cmd, filename = data.decode().split() # 分割接收到的消息,提取出文件名
 if os.path.isfile(filename): # 判断文件是否存在
  f = open(filename, 'rb') # 打开文件
  m = hashlib.md5() # md5加密对象
  file_size = os.stat(filename).st_size # 计算文件大小
  conn.send(str(file_size).encode("utf-8")) # send file size
  conn.recv(1024) # wait the ack 预防粘包
  for line in f:
  m.update(line) # 加密
  conn.send(line) # 发送
  print("file md5:", m.hexdigest()) # 以16进制打印加密后的文件
  f.close() # 关闭文件
  conn.send(m.hexdigest().encode("utf-8")) # 以16进制发送加密后的文件
 print('send done')
socket.close()

服务器的具体实现逻辑和前一篇大体上没什么区别,主要处理细节诶变成了查找文件处理文件和加密文件

3、测试效果

启动服务器

xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py

启动客户端

client

xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_client.py

server

xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py
等待接收文件名...

客户端发送想要接收的文件名

client

xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_client.py
get a.txt
server response: b'90'
file recv done 90 90
md5 2232897a127542c1d0715e66e7ca57cc b'2232897a127542c1d0715e66e7ca57cc'

server

xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py
等待接收文件名...
file md5: 2232897a127542c1d0715e66e7ca57cc
send done
等待接收文件名...

上面的服务器端在发送完数据之后,接着又发送了md5的加密文件,可能会发生粘包的情况,所以上面还存在改进的地方,将预防粘包的代码加进去就比较完美了。

总结

广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。