如何解决客户端与服务端多次收发消息
服务端
import socket
import time
serverSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(serverSocket)
serverSocket.bind(('127.0.0.1', 8080))
serverSocket.listen(5)
print('start')
conn, client_addr = serverSocket.accept()
print(conn)
print(client_addr)
while True:
try:
data = conn.recv(1024)
if data == b'':
break
print('收到客户端的数据', data)
conn.send(data.upper())
except ConnectionResetError as e:
print('------')
break
conn.close()
serverSocket.close()
客户端
import socket
import time
socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(socket)
socket.connect(('127.0.0.1', 8080))
for i in range(0, 20):
socket.send(('hello%s' % (i)).encode('utf-8'))
data = socket.recv(1024)
print('收到服务端的数据', data)
socket.close()
如何一个服务端支持多个客户端
服务端
import socket
import time
serverSocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(serverSocket)
serverSocket.bind(('127.0.0.1', 8080))
serverSocket.listen(5)
print('start')
while True:
conn, client_addr = serverSocket.accept()
print(conn)
print(client_addr)
while True:
try:
data = conn.recv(1024)
if data == b'':
break
print('收到客户端的数据', data)
conn.send(data.upper())
except ConnectionResetError as e:
print('------')
break
conn.close()
serverSocket.close()
客户端
不用改动。多启动接个客户端来处理
带来另一个问题。发现第二次的连接必须等第一次连接完全结束了服务端才能继续处理。这肯定不合适
服务端并发处理客户端连接
通过单线程(协程)来解决
服务端
import socket
from gevent import monkey
monkey.patch_all()
from gevent import spawn
def recv_send(conn):
while True:
try:
data = conn.recv(1024)
if data == b'':
break
print('收到客户端数据', data)
conn.send(data.upper())
except ConnectionResetError:
break
def connection(ip, port, backlog=5):
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind((ip, port))
server.listen(backlog)
while True:
conn, client_addr = server.accept()
spawn(recv_send, conn)
if __name__ == '__main__':
g1 = spawn(connection, '127.0.0.1', 8080)
g1.join()
通过多线程来执行
import socket
from concurrent.futures import ThreadPoolExecutor
from threading import Thread
def recv_send(conn):
while True:
try:
data = conn.recv(1024)
if data == b'':
break
print('收到客户端数据', data)
conn.send(data.upper())
except ConnectionResetError:
break
def connection(ip, port, backlog=5):
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind((ip, port))
server.listen(backlog)
global allconn
while True:
conn, client_addr = server.accept()
t2 = Thread(target=recv_send, args=(conn,))
t2.start()
if __name__ == '__main__':
t1 = Thread(target=connection, args=('127.0.0.1', 8080))
t1.start()
t1.join()
通过 tcpsocketserver 服务器
import socketserver
class MyTCPhanler(socketserver.BaseRequestHandler):
def handle(self):
while True:
try:
data = self.request.recv(1024)
print('收到客户端数据', data)
self.request.send(data.upper())
except ConnectionResetError:
break
if __name__ == '__main__':
server=socketserver.ThreadingTCPServer(('127.0.0.1',8080),MyTCPhanler)
server.serve_forever()
多进程没有意义 参考多线程实现就可
实际案例(远程在服务器上执行命令)
subprocess
'''
1. 什么是进程
进程指的是一个程序的运行过程,或者说一个正在执行的程序
所以说进程一种虚拟的概念,该虚拟概念起源操作系统
subprocess模块
sub 子
process 进程
正在进行中的程序 每当打开一个程序就会开启一个进程
每个进程包含运行程序所需的所有资源
正常情况下 不可以跨进程访问数据 qq不能访问微信,微信不能访问qq
但是有些情况写就需要访问别的进程数据 美团跳转到支付宝 这里跨进程了
提供一个叫做管道的对象 专门用于跨进程通讯
作用:用于执行系统命令
总结 subprocess的好处是可以获取指令的执行结果
'''
import subprocess
cmd = input('输入命令')
obj=subprocess.Popen(cmd,
shell=True,
stdout= subprocess.PIPE,
stderr= subprocess.PIPE,
)
stdout=obj.stdout.read().decode('gbk')
stderr=obj.stderr.read().decode('gbk')
print(stdout+stderr)
服务端
import socket
import subprocess
from threading import Thread
def recv_send(conn):
while True:
try:
cmd = conn.recv(1024)
if cmd == b'':
break
print('收到客户端数据', cmd.decode('utf-8'))
obj = subprocess.Popen(cmd,
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
)
stdout = obj.stdout.read()
stderr = obj.stderr.read()
conn.send(stdout + stderr)
except ConnectionResetError:
break
def connection(ip, port, backlog=5):
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind((ip, port))
server.listen(backlog)
while True:
conn, client_addr = server.accept()
t2 = Thread(target=recv_send, args=(conn,))
t2.start()
if __name__ == '__main__':
t1 = Thread(target=connection, args=('127.0.0.1', 8080))
t1.start()
t1.join()
客户端
import socket
socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(socket)
socket.connect(('127.0.0.1', 8080))
while True:
cmd = input('请输入命令,输入quit退出: ')
if cmd == 'quit':
break
socket.send(cmd.encode('utf-8'))
out = socket.recv(1024)
print('命令执行结果:')
print(out.decode('utf-8'))
socket.close()
粘包问题
上面的例子每次接受指定1024个字节,那如果传输的数据超过了1024个字节怎么办? 如果接受的数据没有读完,下次再读取的时候就会接着上次的读取,这样的话就会导致数据不完整。多了或者少了。我们无法评估网络传输的数据量到底有多少,也不可能传输的数据量有多少我们就在内存中开辟多大的空间。
粘包案例
客户端发送两次 10个字节 服务端每次只接受9个字节 导致服务端第一次只能接受9个字节 剩下一个字节在第二次循环的时候继续接受。 同理第二次循环也没有读完,继续留给第三次循环接受 但是客户端只发送了两次。
服务端
import socket
import subprocess
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(('127.0.0.1', 8080))
server.listen(5)
conn, client_addr = server.accept()
while True:
try:
data = conn.recv(9)
if data == b'':
break
print('收到客户端数据', data.decode('utf-8'))
except ConnectionResetError:
break
客户端
import socket
socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(socket)
socket.connect(('127.0.0.1', 8080))
send_data = '0123456789'
socket.send(send_data.encode('utf-8'))
send_data = '0123456789'
socket.send(send_data.encode('utf-8'))
socket.close()
解决粘包问题
粘包问题是tcp协议流式传输数据的方式导致的 如何解决粘包问题:接收端能够精确地收干净每个数据包没有任何残留 所以客户端必须告诉服务端发送多少数据,就要有个报头
分析: 报头的信息应该包含长度信息 问题1:发送的时候只能是Bytes类型 描述长度的是数字 所以我们应该把整型数字转换成固定长度的Bytes类型
服务端
import socket
import struct
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server.bind(('127.0.0.1', 8080))
server.listen(5)
conn, client_addr = server.accept()
while True:
try:
header = conn.recv(4)
if header == b'':
break
data_len = struct.unpack('i', header)[0]
data = conn.recv(data_len)
print('收到客户端数据', data.decode('utf-8'))
except ConnectionResetError:
break
conn.close()
server.close()
客户端
import socket
import struct
socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
print(socket)
socket.connect(('127.0.0.1', 8080))
send_data = '0123456789'
send_header = struct.pack('i', len(send_data))
socket.send(send_header + send_data.encode('utf-8'))
send_data = '0123456789'
send_header = struct.pack('i', len(send_data))
socket.send(send_header + send_data.encode('utf-8'))
socket.close()
UDP通讯
服务端
import socket
server = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
server.bind(('127.0.0.1', 8080))
while True:
data, client_addr = server.recvfrom(1024)
print(data)
print(client_addr)
server.sendto(data.upper(), client_addr)
客户端
import socket
client = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
while True:
msg = input('>>>.').strip()
client.sendto(msg.encode('utf-8'),('127.0.0.1',8080))
data, server_addr = client.recvfrom(1024)
print(data)
socketserver 模块实现多线程并发服务端
服务端
import socketserver
import time
class MyUphander(socketserver.BaseRequestHandler):
def handle(self):
data, sock = self.request
print(data)
time.sleep(1)
sock.sendto(data.upper(), self.client_address)
if __name__ == '__main__':
server = socketserver.ThreadingUDPServer(('127.0.0.1', 8080), MyUphander)
server.serve_forever()
|