网络编程
位置:首页>> 网络编程>> Python编程>> Python Socket传输文件示例

Python Socket传输文件示例

作者:爱做梦的鱼  发布时间:2023-10-18 17:19:00 

标签:python,socket

发送端可以不停的发送新文件,接收端可以不停的接收新文件。

例如:发送端输入:e:\visio.rar,接收端会默认保存为 e:\new_visio.rar,支持多并发,具体实现如下;

接收端:

方法一:


#-*- coding: UTF-8 -*-
import socket,time,SocketServer,struct,os,thread
host='192.168.50.74'
port=12307
s=socket.socket(socket.AF_INET,socket.SOCK_STREAM) #定义socket类型
s.bind((host,port)) #绑定需要监听的Ip和端口号,tuple格式
s.listen(1)

def conn_thread(connection,address):
 while True:
   try:
     connection.settimeout(600)
     fileinfo_size=struct.calcsize('128sl')
     buf = connection.recv(fileinfo_size)
     if buf: #如果不加这个if,第一个文件传输完成后会自动走到下一句
       filename,filesize =struct.unpack('128sl',buf)
       filename_f = filename.strip('\00')
       filenewname = os.path.join('e:\\',('new_'+ filename_f))
       print 'file new name is %s, filesize is %s' %(filenewname,filesize)
       recvd_size = 0 #定义接收了的文件大小
       file = open(filenewname,'wb')
       print 'stat receiving...'
       while not recvd_size == filesize:
         if filesize - recvd_size > 1024:
           rdata = connection.recv(1024)
           recvd_size += len(rdata)
         else:
           rdata = connection.recv(filesize - recvd_size)
           recvd_size = filesize
         file.write(rdata)
       file.close()
       print 'receive done'
       #connection.close()
   except socket.timeout:
     connection.close()

while True:
 connection,address=s.accept()
 print('Connected by ',address)
 #thread = threading.Thread(target=conn_thread,args=(connection,address)) #使用threading也可以
 #thread.start()
 thread.start_new_thread(conn_thread,(connection,address))

s.close()

方法二:


#-*- coding: UTF-8 -*-
import socket,time,SocketServer,struct,os
host='192.168.50.74'
port=12307
ADDR=(host,port)

class MyRequestHandler(SocketServer.BaseRequestHandler):  
 def handle(self):  
   print('connected from:', self.client_address)
   while True:
     fileinfo_size=struct.calcsize('128sl') #定义文件信息。128s表示文件名为128bytes长,l表示一个int或log文件类型,在此为文件大小
     self.buf = self.request.recv(fileinfo_size)
     if self.buf: #如果不加这个if,第一个文件传输完成后会自动走到下一句
       self.filename,self.filesize =struct.unpack('128sl',self.buf) #根据128sl解包文件信息,与client端的打包规则相同
       print 'filesize is: ',self.filesize,'filename size is: ',len(self.filename) #文件名长度为128,大于文件名实际长度
       self.filenewname = os.path.join('e:\\',('new_'+ self.filename).strip('\00')) #使用strip()删除打包时附加的多余空字符
       print self.filenewname,type(self.filenewname)
       recvd_size = 0 #定义接收了的文件大小
       file = open(self.filenewname,'wb')
       print 'stat receiving...'
       while not recvd_size == self.filesize:
         if self.filesize - recvd_size > 1024:
           rdata = self.request.recv(1024)
           recvd_size += len(rdata)
         else:
           rdata = self.request.recv(self.filesize - recvd_size)
           recvd_size = self.filesize
         file.write(rdata)
       file.close()
       print 'receive done'
   #self.request.close()

tcpServ = SocketServer.ThreadingTCPServer(ADDR, MyRequestHandler)
print('waiting for connection...' )
tcpServ.serve_forever()

发送端:


#-*- coding: UTF-8 -*-
import socket,os,struct
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.connect(('192.168.50.74',12307))
while True:

filepath = raw_input('Please Enter chars:\r\n')
 if os.path.isfile(filepath):
   fileinfo_size=struct.calcsize('128sl') #定义打包规则
   #定义文件头信息,包含文件名和文件大小
   fhead = struct.pack('128sl',os.path.basename(filepath),os.stat(filepath).st_size)
   s.send(fhead)
   print 'client filepath: ',filepath
   # with open(filepath,'rb') as fo: 这样发送文件有问题,发送完成后还会发一些东西过去
   fo = open(filepath,'rb')
   while True:
     filedata = fo.read(1024)
     if not filedata:
       break
     s.send(filedata)
   fo.close()
   print 'send over...'
   #s.close()

来源:http://www.cnblogs.com/dreamer-fish/p/5501924.html

0
投稿

猜你喜欢

手机版 网络编程 asp之家 www.aspxhome.com