Fork me on GitHub

基于tcp的下载文件,以及struct模块的应用。

一 基于TCP的下载

客户端:

from socket import *
import os
def main():
    tcp_socket = socket(AF_INET, SOCK_STREAM)# 建立套接字
    tcp_ip = input("请输入ip:")
    tcp_port = int(input("请输入端口:"))# 接收用输入的服务器端的ip和端口
    tcp_socket.connect((tcp_ip, tcp_port))# 连接服务器
    file_name = input("请输入要下载的文件名:")# 输入要下载的文件名
    tcp_socket.send(file_name.encode())# 将文件名发送至服务器端
    new_file = open(file_name, "wb")# 创建一个空文件
    time = 0# 用于计算读取的字节数
    while True:
        mes = tcp_socket.recv(1024)# 接收服务器端返回的内容
        if mes:# 如果内容不为空执行
            new_file.write(mes.decode())# 解码并向文件内写入
            time += len(mes)# 计算字节
        else:
            if time == 0:# 如果字节数为空即未收到内容
                new_file.close()# 关闭文件
                os.remove(file_name) # 删除刚刚创建的文件
                print("没有您要下载的文件")
            else:
                print("文件下载成功") # 如过time有值时name文件传输完成
            break
    tcp_socket.close() # 关闭套接字
if __name__ == '__main__':
 main()

服务端:

import socket
def file_deal(file_name):# 定义函数用于处理用户索要下载的文件
    try:
        files = open(file_name, "rb") # 二进制方式读取
        mes = files.read()
    except:
        print("没有该文件")
    else:
        files.close()
        return mes
def main():
    tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)# 创建套接字
    tcp_socket.bind(("",8080)) # 固定端口号
    tcp_socket.listen(128)# 将主动套接字转为被动套接字,监听连接客户端
    while True:
        client_socket,client_addr = tcp_socket.accept()# 利用accept获取分套接字以及客户端的地址
        file_name = client_socket.recv(4096)# 接收客户端的数据
        mes = file_deal(file_name)# 调用函数处理用户下载的文件
        if mes:# 如果文件不为空发送
            client_socket.send(mes)
        client_socket.close()#关闭分套接字
if __name__ == "__main__":
    main()

 二  基于TCP的套接字通讯:

客户端client_side:

import socket c
=socket.socket(socket.AF_INET,socket.SOCK_STREAM)      #套接字 c.connect(('127.0.0.1',8080))                  #connect对应server_side中的accept。 while True: try: msg=input('>:') c.send(msg.encode()) s_data=c.recv(1024) print(s_data.decode()) except Exception as e: print(e) c.close()

服务端server_side:
import socket

s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.bind(('127.0.0.1',8080))
s.listen(5)

while True:
conn,addr=s.accept()#等待客户端的连接,对应client_side端的connect。
try:
while True:
client_data=conn.recv(1024)
conn.send(client_data.upper())
print(client_data.decode())
except Exception as e:
print(e)
conn.close()

s.close()
TCP的一个通讯过程,服务端先启动,通过导入socket模块,产生一个客户端套接字,首先我们应该绑定一个固定的ip与端口port,那么客户端再访问就不需要每次都去找
服务端。那么套接字下面的listen()功能,主要是限制同时请求数,也就是listen控制的是
同时请求数,而不是连接数,连接数是跟服务器的性能相关的,
同时请求数,而不是连接数,连接数是跟服务器的性能相关的,性能越好能建的连接数也越多,这些都准备好之后就是等待客户端的连接请求,我们这里然后再accept()
这里等待客户端连接进来,那么accept()其实就对应于客户端的connect(),那么一旦连接建成,就准备进行数据的传输,我们客户端首先发一个数据请求,其实客户端的
数据首先是send()到客户端的一个操作系统的缓存中,因为应用软件是没办法直接操作网卡进行数据的传输,所以他要借助操作系统来调用网卡,进行数据的传输,
那么再send()数据的时候我们pycharm会有一个优化的机制nagle,就是对于数据量小且发送间隔短的数据会一起发送给服务端,这是造成粘包的一种情况。那么我们在接收
的时候同样是从操作系统的缓存中取数据,那么每次如果最大值是recv(1024)的话,就会造成可能一次没办法取出来全部的发送过来的数据,同样造成粘包。

三 基于UDP的套接字通讯:

server_side
import socket s
=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) s.bind(('127.0.0.1',8080)) while True: data,addr=s.recvfrom(1024) s.sendto(data.upper(),addr) print(data,addr) s.close()

client_side:
import socket
c=socket.socket(socket.AF_INET,socket.SOCK_DGRAM)

while True:

c.sendto(b'hello',('127.0.0.1',8080))
c.sendto(b'world',('127.0.0.1',8080))
c.sendto(b'python',('127.0.0.1',8080))
data,addr=c.recvfrom(1024)
print(data.decode(),addr)
c.close()
 

 

posted @ 2018-05-01 18:13  道阻切长  阅读(181)  评论(0编辑  收藏  举报