一 基于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',))                  #connect对应server_side中的accept。
while True:
try:
msg=input('>:')
c.send(msg.encode())
s_data=c.recv()
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',)) while True:
data,addr=s.recvfrom()
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()
 
04-18 12:42
查看更多