客户端向服务器发送一个请求,请求内容是一个文件名,服务器在查找自己这边有没有这个文件,如果有的话就发送给客户端
1、客户端
- 生成socket对象
- 建立连接
- 输入想要接收的文件
- 将输入的文件名发送给服务器
- 接收服务器发回的关于即将要发送来的文件的大小
- 发送一条信息给服务器告诉它准备好接收了
- 接收文件数据
- 打印全部接收的提示信息
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、服务器
- 生成socket连接对象
- 绑定要监听端口
- 监听
- 等待客户端的连接
- 接收客户端发来的文件名
- 查找这个文件是否存在
- 如果存在,打开文件计算大小,发送给客户端
- 等待客户端确认
- 发送文件具体内容给客户端
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的加密文件,可能会发生粘包的情况,所以上面还存在改进的地方,将预防粘包的代码加进去就比较完美了。
总结
更新日志
- 雨林唱片《赏》新曲+精选集SACD版[ISO][2.3G]
- 罗大佑与OK男女合唱团.1995-再会吧!素兰【音乐工厂】【WAV+CUE】
- 草蜢.1993-宝贝对不起(国)【宝丽金】【WAV+CUE】
- 杨培安.2009-抒·情(EP)【擎天娱乐】【WAV+CUE】
- 周慧敏《EndlessDream》[WAV+CUE]
- 彭芳《纯色角3》2007[WAV+CUE]
- 江志丰2008-今生为你[豪记][WAV+CUE]
- 罗大佑1994《恋曲2000》音乐工厂[WAV+CUE][1G]
- 群星《一首歌一个故事》赵英俊某些作品重唱企划[FLAC分轨][1G]
- 群星《网易云英文歌曲播放量TOP100》[MP3][1G]
- 方大同.2024-梦想家TheDreamer【赋音乐】【FLAC分轨】
- 李慧珍.2007-爱死了【华谊兄弟】【WAV+CUE】
- 王大文.2019-国际太空站【环球】【FLAC分轨】
- 群星《2022超好听的十倍音质网络歌曲(163)》U盘音乐[WAV分轨][1.1G]
- 童丽《啼笑姻缘》头版限量编号24K金碟[低速原抓WAV+CUE][1.1G]