36、基于TCP、UDP协议的嵌套字通信
一、套接字工作流程
1.1、套接字流程
服务器端先初始化socket:(创建套接字模块)
和端口绑定bind:(将地址绑定到套接字)
对端口进行监听listen:(监听链接)
调用accept堵塞:(接收服务器链接)
等待客户端连接connect:(客户端尝试连接服务器)
客户端发送请求send:(客户端发出请求)
服务器接收请求并处理请求recv:(服务器接收请求)
将数据返回给客户端send:(服务器发出请求)
客户端读取数据recv:(客户端接收请求)
关闭连接close(关闭服务器)
一、基于TCP协议的嵌套字通信
TCP是基于链接的,所以需要先启动服务端,在启动客户端去连接服务端
1.1、名词定义
SOCK_STREAM:流式协议(tcp协议)
AF_INET:网络通讯协议
0.0.0.0:公网服务器地址
127.0.0.1:服务器地址(自己的,通常用来调试)
二、通讯循环
由于tcp是基于链接通信的,所以一旦程序执行完毕就会直接断开,所以需要在程序内添加while循环,保证可以持续服务
2.1、问题点
由于tcp是基于链接通信的,连接的双方,所以:
1.一旦客户端单方向断开连接,服务器拿到的conn就是无效的,服务器在Windows上会直接报错,linux上会直接死循环
2.客户端发送的信息为空,应用程序会将信息发给操作系统,操作系统不会将空往外发送,因此,服务器就不会收到消息,而客户端则在等待消息
2.2、解决方案
1.使用异常捕获,捕获后中断程序
2.使用if判断,如果收到的消息为空,则跳出循环
三、连接循环
服务器的终极目标就是永不停机,所以需要将客户端连接服务器加入while
3.1、问题
端口占用:一个端口只能同时执行一个客户端,因此前一个未停止,下一个无法执行,在半连接池里的客户端可以输入内容排队等候
accept限制了服务器连接客户端的数量,一旦超过将不会进入半连接池
3.2、解决方法
phone=socket(AF_INET,SOCK_STREAM) phone.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加 phone.bind((‘127.0.0.1‘,8080))
服务端
# 服务端应该满足的特点:
# 1、一直提供服务
# 2、并发地提供服务
import socket
# 1、买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) # 流式协议=》tcp协议
# 2、绑定手机卡
phone.bind((‘127.0.0.1‘,8080)) # 0-65535, 1024以前的都被系统保留使用
# 3、开机
phone.listen(5) # 5指的是半连接池的大小
print(‘服务端启动完成,监听地址为:%s:%s‘ %(‘127.0.0.1‘,8080))
# 4、等待电话连接请求:拿到电话连接conn
# 加上链接循环
while True:
conn,client_addr=phone.accept()
# 5、通信:收\发消息
while True:
try:
data=conn.recv(1024) # 最大接收的数据量为1024Bytes,收到的是bytes类型
if len(data) == 0:
# 在unix系统洗,一旦data收到的是空
# 意味着是一种异常的行为:客户度非法断开了链接
break
print("客户端发来的消息:",data.decode(‘utf-8‘))
conn.send(data.upper())
except Exception:
# 针对windows系统
break
# 6、关闭电话连接conn(必选的回收资源的操作)
conn.close()
# 7、关机(可选操作)
phone.close()客户端
import socket
#1、买手机
phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) # 流式协议=》tcp协议
#2、拨通服务端电话
phone.connect((‘127.0.0.1‘,8080))
#3、通信
while True:
msg=input("输入要发送的消息>>>: ").strip() #msg=‘‘
if len(msg) == 0:continue
phone.send(msg.encode(‘utf-8‘))
print(‘======?‘)
data=phone.recv(1024)
print(data.decode(‘utf-8‘))
#4、关闭连接(必选的回收资源的操作)
phone.close()四、基于udp协议的套接字通信
udp是无链接的,所以无论哪一端都不会报错
服务器和客户端需要初始化socket,设定连接地址,接收数据,以及发出数据,最后断开连接
服务器
import socket
server=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) # 数据报协议=》udp协议
server.bind((‘127.0.0.1‘,8081)) # IP地址
while True:
data,client_addr=server.recvfrom(1024) #接收信息
server.sendto(data.upper(),client_addr) #发出信息
server.close()客户端
import socket
client=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) # 流式协议=》tcp协议
while True:
msg=input(‘>>>: ‘).strip()
client.sendto(msg.encode(‘utf-8‘),(‘127.0.0.1‘,8081)) #发出信息,设置地址
res=client.recvfrom(1024) #接收信息
print(res)
client.close()