python 實(shí)現(xiàn)socket服務(wù)端并發(fā)的四種方式
多進(jìn)程&多線程
服務(wù)端:多進(jìn)程和多線程的開啟方式相同。
缺點(diǎn):<1> 由于Cpython的GIL,導(dǎo)致同一時(shí)間無法運(yùn)行多個(gè)線程;<2> 不可能無限開進(jìn)進(jìn)程或線程
解決辦法:多進(jìn)程、concurrent.futures.ProcessPoolExecutor、線程池
import socket
from multiprocessing import Process
from threading import Thread
class MyTcpServer:
def __init__(self, ip, port):
self.ip = ip
self.port = port
self.server = socket.socket()
self.server.bind((self.ip, self.port))
self.server.listen(5)
def wait_accept(self):
conn, addr = self.server.accept()
return conn, addr
def handle_request(self, conn):
while 1:
try:
data = conn.recv(1024)
if not data: break
conn.send(data.upper())
except Exception as e:
print(e)
break
conn.close()
if __name__ == '__main__':
server = MyTcpServer('127.0.0.1', 8888)
while 1:
conn, addr = server.wait_accept()
p = Process(target=server.handle_request, args=(conn, )) # 創(chuàng)建一個(gè)進(jìn)程
p.start() # 告訴操作提供,開啟這個(gè)進(jìn)程
進(jìn)程池&線程池
異步提交任務(wù),支持異步接收返回結(jié)果(submit返回一個(gè)futures對象,調(diào)用add_done_callback方法)
import socket
from concurrent.futures import ProcessPoolExecutor
# from concurrent.futures import ThreadPoolExecutor
class MyTcpServer:
def __init__(self, ip, port):
self.ip = ip
self.port = port
self.server = socket.socket()
self.server.bind((self.ip, self.port))
self.server.listen(5)
def wait_accept(self):
conn, addr = self.server.accept()
return conn, addr
def handle_request(self, conn):
while 1:
try:
data = conn.recv(1024)
if not data: break
conn.send(data.upper())
except Exception as e:
print(e)
break
conn.close()
if __name__ == '__main__':
server = MyTcpServer('127.0.0.1', 8888)
pool = ProcessPoolExecutor(5) # 5個(gè)進(jìn)程一直服務(wù)
while 1:
conn, addr = server.wait_accept()
pool.submit(server.handle_request, conn) # 異步提交任務(wù)
socketserver
優(yōu)點(diǎn):簡化socket服務(wù)端創(chuàng)建流程。
提供服務(wù)端串行和并發(fā)兩種服務(wù)模式(TCPServer,ThreadingTCPServer)
缺點(diǎn):windows上無法使用多進(jìn)程實(shí)現(xiàn)并發(fā)
import socketserver
class MyTcpHandler(socketserver.BaseRequestHandler):
def handle(self): # 通信循環(huán)
while 1:
try:
data = self.request.recv(1024)
if not data: break
self.request.send(data.upper())
except Exception as e:
print(e)
break
self.request.close()
if __name__ == '__main__':
ip_port = '127.0.0.1', 8888
server = socketserver.ThreadingTCPServer(ip_port, MyTcpHandler) # 異步處理
server.serve_forever() # 連接循環(huán)
協(xié)程
優(yōu)點(diǎn):單線程內(nèi)實(shí)現(xiàn)并發(fā),代碼級別模擬IO切換,提高程序運(yùn)行效率
from gevent import spawn, monkey;monkey.patch_all() # 猴子補(bǔ)丁,補(bǔ)丁:常規(guī)IO
import socket
class MyTcpServer:
def __init__(self, ip, port, my_spawn):
self.ip = ip
self.port = port
self.server = socket.socket()
self.server.bind((self.ip, self.port))
self.server.listen(5)
self.spawn = my_spawn # 保存spawn本地
def wait_accept(self):
while 1:
conn, addr = self.server.accept()
self.spawn(self.handle_request, conn) # 檢測 handle_request的io
def handle_request(self, conn):
while 1:
try:
data = conn.recv(1024)
if not data: break
conn.send(data.upper())
except Exception as e:
print(e)
break
conn.close()
if __name__ == '__main__':
server = MyTcpServer('127.0.0.1', 8888, spawn)
g1 = server.spawn(server.wait_accept) # 檢測wait_accept的io
g1.join() # 等待g1運(yùn)行結(jié)束,即一直在循環(huán)檢測io
以上就是python 實(shí)現(xiàn)socket服務(wù)端并發(fā)的四種方式的詳細(xì)內(nèi)容,更多關(guān)于python socket服務(wù)端并發(fā)的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python3操作SQL Server數(shù)據(jù)庫(實(shí)例講解)
下面小編就為大家?guī)硪黄狿ython3操作SQL Server數(shù)據(jù)庫(實(shí)例講解)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-10-10
PyPDF2讀取PDF文件內(nèi)容保存到本地TXT實(shí)例
這篇文章主要介紹了PyPDF2讀取PDF文件內(nèi)容保存到本地TXT實(shí)例,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-05-05
Python后臺開發(fā)Django的教程詳解(啟動(dòng))
這篇文章主要介紹了Python后臺開發(fā)Django(啟動(dòng)),本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2019-04-04
python中for循環(huán)的多種使用實(shí)例
for語句是Python中執(zhí)行迭代的兩個(gè)語句之一,另一個(gè)語句是while,下面這篇文章主要給大家介紹了關(guān)于python中for循環(huán)的多種使用方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-09-09

