python實(shí)現(xiàn)ftp文件傳輸功能
本文實(shí)例為大家分享了python實(shí)現(xiàn)ftp文件傳輸?shù)木唧w代碼,供大家參考,具體內(nèi)容如下
主要步驟可以分為以下幾步:
1.讀取文件名
2.檢測(cè)文件是否存在
3.打開(kāi)文件
4.檢測(cè)文件大小
5.發(fā)送文件大小和 md5值給客戶端
6.等客戶端確認(rèn)
7.開(kāi)始邊讀邊發(fā)數(shù)據(jù)
服務(wù)器端代碼:
import socket,os,time
import hashlib
server =socket.socket()
server.bind(('0.0.0.0',6666))
server.listen()
print("等待....")
while True:
conn,addr = server.accept()
print("new conn:",conn)
while True:
data = conn.recv(1024)
if not data:
print("client is disconnection")
break
cmd,filename = data.decode().split() #記錄指令和文件名
print(filename)
#判斷當(dāng)前目錄是否存在該文件,而且必須是文件,而不是目錄
if os.path.isfile(filename):
f = open(filename,'rb')
#m = hashlib.md5() # 創(chuàng)建md5
file_size = os.stat(filename).st_size #stat() 可以返回文件的大小值
conn.send((str(file_size)).encode()) # 發(fā)送文件大小
conn.recv(1024) #等待返回信息
for line in f:
# m.updata(line)
conn.send(line)
#print("file md5",m.hexdigest()) #打印md5值
f.close()
客戶端代碼:
# Author: zjt
import socket
client = socket.socket()
client.connect(("0.0.0.0",6666))
while True:
cmd = input(">>>:").strip()
if len(cmd)==0 :continue
if cmd.startswith("get"):
client.send(cmd.encode())
server_response = client.recv(1024)
print("server response: ",server_response)
client.send(b"ready to recv file")
# 開(kāi)始接收文件
file_total_size = int(server_response.decode())
received_size = 0 # 記錄接收文件的大小
filename = cmd.split()[1]
# 因?yàn)閮蓚€(gè)目錄一致,接收的文件名不能與原文件相同
f = open(filename+".new","wb")
while received_size < file_total_size:
data = client.recv(1024)
received_size += len(data)
f.write(data)
print("total:",file_total_size," present: ",received_size)
else:
print("file has received done!")
f.close()
client.close()
用80M的文件傳輸測(cè)試,效果如下:

程序升級(jí):
前面的代碼還沒(méi)添加md5進(jìn)行驗(yàn)證,現(xiàn)在對(duì)代碼進(jìn)行升級(jí)
服務(wù)器端代碼:
import socket,os,time
import hashlib
server =socket.socket()
server.bind(('0.0.0.0',8888))
server.listen()
print("等待....")
while True:
conn,addr = server.accept()
print("new conn:",conn)
while True:
data = conn.recv(1024)
if not data:
print("client is disconnection")
break
cmd,filename = data.decode().split() #記錄指令和文件名
print(filename)
#判斷當(dāng)前目錄是否存在該文件,而且必須是文件,而不是目錄
if os.path.isfile(filename):
f = open(filename,'rb')
m = hashlib.md5() # 創(chuàng)建md5
file_size = os.stat(filename).st_size #stat() 可以返回文件的大小值
conn.send((str(file_size)).encode()) # 發(fā)送文件大小
conn.recv(1024) #等待返回信息
for line in f:
m.update(line)
conn.send(line)
print("file md5",m.hexdigest()) #打印md5值
f.close()
conn.send(m.hexdigest().encode()) # 發(fā)送md5
print("我真的已經(jīng)發(fā)過(guò)去了",m.hexdigest().encode())
print("send done")
server.close()
客戶端代碼:
import socket
import hashlib
client = socket.socket()
client.connect(("0.0.0.0",8888))
while True:
cmd = input(">>>:").strip()
if len(cmd)==0 :continue
if cmd.startswith("get"):
client.send(cmd.encode())
server_response = client.recv(1024)
print("server response: ",server_response)
client.send(b"ready to recv file")
# 開(kāi)始接收文件
file_total_size = int(server_response.decode())
received_size = 0 # 記錄接收文件的大小
filename = cmd.split()[1]
# 因?yàn)閮蓚€(gè)目錄一致,接收的文件名不能與原文件相同
f = open(filename+".new","wb")
m = hashlib.md5()
while received_size < file_total_size:
data = client.recv(1024)
received_size += len(data)
m.update(data)
f.write(data)
#print("total:",file_total_size," present: ",received_size)
else:
new_file_md5 = m.hexdigest()
print("client file md5:",new_file_md5)
print("file has received done!")
print("total:",file_total_size," present: ",received_size)
f.close()
sever_file_md5 = client.recv(1024)
print("client file md5:",new_file_md5)
print("server file md5:",sever_file_md5)
client.close()
兩個(gè)程序在linux 環(huán)境下運(yùn)行,結(jié)果如下:

可以看到傳輸后文件大小變大了一點(diǎn)點(diǎn),而且md5前后值也不同,說(shuō)明文件傳輸發(fā)生了改變。
現(xiàn)在講程序在windows環(huán)境下運(yùn)行,結(jié)果如下:

此時(shí)可以看到windows上沒(méi)有問(wèn)題,文件大小相同,且md5值也一致。
原因分析:
之所以會(huì)發(fā)生這種情況,是因?yàn)樵趌inux上運(yùn)行時(shí),最后一次傳輸文件與發(fā)送md5值的時(shí)候,發(fā)生可粘包,導(dǎo)致最后一次接收文件的時(shí)候,連同md5的數(shù)據(jù)一并發(fā)送了。而客戶端也當(dāng)作一條接收信息,全部接收了。所以客戶端出現(xiàn)沒(méi)有收到來(lái)自服務(wù)器端的md5值,多出來(lái)的那一點(diǎn)點(diǎn),就是md5值的大小。
解決方法:
在接收文件的時(shí)候,判斷當(dāng)前剩余多少文件需要接收,如果大于1024,就接收1024大小的文件,否則就只接收剩下全部的文件,防止最后一次接收多余的數(shù)據(jù)。
只需要對(duì)客戶端代碼進(jìn)行修改,修改后代碼如下:
import socket
import hashlib
client = socket.socket()
client.connect(("0.0.0.0",8888))
while True:
cmd = input(">>>:").strip()
if len(cmd)==0 :continue
if cmd.startswith("get"):
client.send(cmd.encode())
server_response = client.recv(1024)
print("server response: ",server_response)
client.send(b"ready to recv file")
# 開(kāi)始接收文件
file_total_size = int(server_response.decode())
received_size = 0 # 記錄接收文件的大小
filename = cmd.split()[1]
f = open(filename+".new","wb")
m = hashlib.md5()
while received_size < file_total_size:
#添加一次判斷,使最后一次剩多少就接收多少,避免發(fā)生粘包
if file_total_size - received_size > 1024:
size = 1024
else: # 最后一次,剩多少收多少
size = file_total_size - received_size
data = client.recv(size)
received_size += len(data)
m.update(data)
f.write(data)
else:
new_file_md5 = m.hexdigest()
print("client file md5:",new_file_md5)
print("file has received done!")
print("total:",file_total_size," present: ",received_size)
print("下一句關(guān)閉文件")
f.close()
print("開(kāi)始接收md5 ")
sever_file_md5 = client.recv(1024)
print("client file md5:",new_file_md5)
print("server file md5:",sever_file_md5)
client.close()
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Python 使用SFTP和FTP實(shí)現(xiàn)對(duì)服務(wù)器的文件下載功能
- python從ftp獲取文件并下載到本地
- Python+pyftpdlib實(shí)現(xiàn)局域網(wǎng)文件互傳
- python實(shí)現(xiàn)從ftp上下載文件的實(shí)例方法
- Python如何實(shí)現(xiàn)FTP功能
- 基于python實(shí)現(xiàn)FTP文件上傳與下載操作(ftp&sftp協(xié)議)
- Python Socketserver實(shí)現(xiàn)FTP文件上傳下載代碼實(shí)例
- python實(shí)現(xiàn)FTP文件傳輸?shù)姆椒ǎǚ?wù)器端和客戶端)
- python實(shí)現(xiàn)ssh及sftp功能(實(shí)例代碼)
- python實(shí)現(xiàn)從ftp服務(wù)器下載文件
- python FTP編程基礎(chǔ)入門
相關(guān)文章
Python時(shí)間處理模塊time和datetime詳解
本文詳細(xì)介紹了Python中常用的時(shí)間處理模塊time和datetime,time模塊提供多種時(shí)間獲取和轉(zhuǎn)換功能,datetime模塊則在time的基礎(chǔ)上增加了日期和時(shí)間的組合處理,如datetime.now()獲取當(dāng)前日期時(shí)間,兩個(gè)模塊在日常編程中非常有用,尤其是在需要時(shí)間日期計(jì)算和轉(zhuǎn)換的場(chǎng)景下2024-10-10
基于Python實(shí)現(xiàn)讀取嵌套壓縮包下文件的方法
工作中遇到的問(wèn)題,需要用Python實(shí)現(xiàn)嵌套壓縮包下文件讀取,本文給大家介紹了詳細(xì)的解決方法,并有相關(guān)的代碼示例供大家參考,需要的朋友可以參考下2025-04-04
Python中從文件路徑中提取文件名及擴(kuò)展名的多種方法
在文件處理或路徑解析過(guò)程中,提取文件名和擴(kuò)展名是一個(gè)常見(jiàn)且基本的操作,本文將介紹如何通過(guò)字符串操作提取文件名(包括擴(kuò)展名)及文件名的部分內(nèi)容,同時(shí)還將討論一些常見(jiàn)的應(yīng)用場(chǎng)景,需要的朋友可以參考下2025-03-03
如何用Python實(shí)現(xiàn)簡(jiǎn)單的Markdown轉(zhuǎn)換器
這篇文章主要介紹了如何用Python實(shí)現(xiàn)簡(jiǎn)單的Markdown轉(zhuǎn)換器,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07
Numpy實(shí)現(xiàn)卷積神經(jīng)網(wǎng)絡(luò)(CNN)的示例
這篇文章主要介紹了Numpy實(shí)現(xiàn)卷積神經(jīng)網(wǎng)絡(luò)(CNN)的示例,幫助大家更好的理解和使用Numpy,感興趣的朋友可以了解下2020-10-10
Python報(bào)錯(cuò)error: subprocess-exited-with-error解決辦法
在Python開(kāi)發(fā)中,遇到subprocess-exited-with-error通常是由依賴缺失、權(quán)限問(wèn)題、環(huán)境配置錯(cuò)誤或兼容性問(wèn)題導(dǎo)致,修復(fù)方法包括安裝依賴、使用虛擬環(huán)境、提升權(quán)限、檢查路徑和命令,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-10-10

