python實(shí)現(xiàn)socket+threading處理多連接的方法
一.運(yùn)行效果
先說(shuō)兩句,之前我在網(wǎng)上找的相關(guān)文章標(biāo)題上寫(xiě)的是處理多連接,尼瑪,全是假的。網(wǎng)上那些,根本不能異步處理多連接,不能主動(dòng)給客戶(hù)端發(fā)消息。

在服務(wù)端控制臺(tái)輸入1,查看在線人數(shù)。

給指定客戶(hù)端發(fā)送消息。

二.開(kāi)發(fā)思路
以下說(shuō)的是服務(wù)端開(kāi)發(fā)思路,客戶(hù)端比較簡(jiǎn)單就不說(shuō)了。
首先,需要明白的是socket的accept和recv這兩個(gè)方法是阻塞線程的。這就意味著我們需要新開(kāi)線程來(lái)處理這兩個(gè)方法。
具體的程序流程大概是這樣的:
1.新開(kāi)一個(gè)線程用于接收新的連接(socket.accept())
2.當(dāng)有新的連接時(shí),再新開(kāi)一個(gè)線程,用于接收這個(gè)連接的消息(socket.recv())
3.主線程做為控制臺(tái),接收用戶(hù)的輸入,進(jìn)行其他操作
也就是說(shuō),服務(wù)端需要為每一個(gè)連接創(chuàng)建一個(gè)線程。
三.服務(wù)端代碼
全局部分的代碼:
import socket # 導(dǎo)入 socket 模塊
from threading import Thread
ADDRESS = ('127.0.0.1', 8712) # 綁定地址
g_socket_server = None # 負(fù)責(zé)監(jiān)聽(tīng)的socket
g_conn_pool = [] # 連接池
其中,g_conn_pool是用來(lái)存放每一個(gè)客戶(hù)端socket的。
初始化服務(wù)端:
def init():
"""
初始化服務(wù)端
"""
global g_socket_server
g_socket_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 創(chuàng)建 socket 對(duì)象
g_socket_server.bind(ADDRESS)
g_socket_server.listen(5) # 最大等待數(shù)(有很多人理解為最大連接數(shù),其實(shí)是錯(cuò)誤的)
print("服務(wù)端已啟動(dòng),等待客戶(hù)端連接...")
socket.listen()的參數(shù)意義不要理解錯(cuò)了哦~
接收客戶(hù)端連接和處理客戶(hù)端消息:
def accept_client():
"""
接收新連接
"""
while True:
client, _ = g_socket_server.accept() # 阻塞,等待客戶(hù)端連接
# 加入連接池
g_conn_pool.append(client)
# 給每個(gè)客戶(hù)端創(chuàng)建一個(gè)獨(dú)立的線程進(jìn)行管理
thread = Thread(target=message_handle, args=(client,))
# 設(shè)置成守護(hù)線程
thread.setDaemon(True)
thread.start()
def message_handle(client):
"""
消息處理
"""
client.sendall("連接服務(wù)器成功!".encode(encoding='utf8'))
while True:
bytes = client.recv(1024)
print("客戶(hù)端消息:", bytes.decode(encoding='utf8'))
if len(bytes) == 0:
client.close()
# 刪除連接
g_conn_pool.remove(client)
print("有一個(gè)客戶(hù)端下線了。")
break
設(shè)置成守護(hù)線程的目的是為了防止主線程退出之后,程序進(jìn)程不退出。
消息處理這一塊,如果接收到一個(gè)長(zhǎng)度為0的數(shù)據(jù)包,那么說(shuō)明客戶(hù)端掉線了。實(shí)際開(kāi)發(fā)中,我們一般采用心跳包機(jī)制判斷客戶(hù)端是否在線。
最后,調(diào)用這些方法,讓服務(wù)端啟動(dòng):
if __name__ == '__main__':
init()
# 新開(kāi)一個(gè)線程,用于接收新連接
thread = Thread(target=accept_client)
thread.setDaemon(True)
thread.start()
# 主線程邏輯
while True:
cmd = input("""--------------------------
輸入1:查看當(dāng)前在線人數(shù)
輸入2:給指定客戶(hù)端發(fā)送消息
輸入3:關(guān)閉服務(wù)端
""")
if cmd == '1':
print("--------------------------")
print("當(dāng)前在線人數(shù):", len(g_conn_pool))
elif cmd == '2':
print("--------------------------")
index, msg = input("請(qǐng)輸入“索引,消息”的形式:").split(",")
g_conn_pool[int(index)].sendall(msg.encode(encoding='utf8'))
elif cmd == '3':
exit()
如果對(duì)流程不清楚的話,可以再回到上面,看看文章的第二部分。
四.客戶(hù)端代碼
客戶(hù)端這部分設(shè)計(jì)的很簡(jiǎn)單。
import socket # 導(dǎo)入 socket 模塊
s = socket.socket() # 創(chuàng)建 socket 對(duì)象
s.connect(('127.0.0.1', 8712))
print(s.recv(1024).decode(encoding='utf8'))
s.send("連接了".encode('utf8'))
print(s.recv(1024).decode(encoding='utf8'))
input("")
最后的input是為了阻塞線程,防止程序接收到消息之后就退出了。
以上這篇python實(shí)現(xiàn)socket+threading處理多連接的方法就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- python使用socket連接遠(yuǎn)程服務(wù)器的方法
- Python socket處理client連接過(guò)程解析
- python通過(guò)socket實(shí)現(xiàn)多個(gè)連接并實(shí)現(xiàn)ssh功能詳解
- python使用socket遠(yuǎn)程連接錯(cuò)誤處理方法
- python使用socket進(jìn)行簡(jiǎn)單網(wǎng)絡(luò)連接的方法
- Python 網(wǎng)絡(luò)編程起步(Socket發(fā)送消息)
- python網(wǎng)絡(luò)編程學(xué)習(xí)筆記(三):socket網(wǎng)絡(luò)服務(wù)器
- Python采用socket模擬TCP通訊的實(shí)現(xiàn)方法
- python實(shí)現(xiàn)socket客戶(hù)端和服務(wù)端簡(jiǎn)單示例
- Python Socket實(shí)現(xiàn)簡(jiǎn)單TCP Server/client功能示例
- python使用socket向客戶(hù)端發(fā)送數(shù)據(jù)的方法
- Python socket連接中的粘包、精確傳輸問(wèn)題實(shí)例分析
相關(guān)文章
Python多元非線性回歸及繪圖的實(shí)現(xiàn)
本文主要介紹了Python多元非線性回歸及繪圖的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2024-04-04
Python自動(dòng)化實(shí)戰(zhàn)之接口請(qǐng)求的實(shí)現(xiàn)
本文為大家重點(diǎn)介紹如何通過(guò) python 編碼來(lái)實(shí)現(xiàn)我們的接口測(cè)試以及通過(guò)Pycharm的實(shí)際應(yīng)用編寫(xiě)一個(gè)簡(jiǎn)單接口測(cè)試,感興趣的可以了解一下2022-05-05
利用Python繪制隨機(jī)游走圖的詳細(xì)過(guò)程
隨機(jī)游走(random walk)也稱(chēng)隨機(jī)漫步,隨機(jī)行走等,是以隨機(jī)的方式采取連續(xù)步驟的過(guò)程,下面這篇文章主要給大家介紹了關(guān)于利用Python繪制隨機(jī)游走圖的相關(guān)資料,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2024-02-02
Python制作簡(jiǎn)單的網(wǎng)頁(yè)爬蟲(chóng)
自己寫(xiě)的一個(gè)爬蟲(chóng),模仿了python核心編程書(shū)里的程序,有詳細(xì)的注釋。 是我一個(gè)理解學(xué)習(xí)的過(guò)程吧。 有需要的小伙伴可以參考下2015-11-11
Python 最大概率法進(jìn)行漢語(yǔ)切分的方法
今天小編就為大家分享一篇Python 最大概率法進(jìn)行漢語(yǔ)切分的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-12-12
Python中子類(lèi)繼承父類(lèi)傳遞參數(shù)的方法
在面向?qū)ο缶幊讨?繼承是一種使得一個(gè)類(lèi)(稱(chēng)為子類(lèi))能夠獲取另一個(gè)類(lèi)(稱(chēng)為父類(lèi))的屬性和方法的機(jī)制,Python支持繼承,并提供了靈活的方式來(lái)繼承和擴(kuò)展類(lèi)的功能,本文將詳細(xì)介紹如何在Python中使用子類(lèi)繼承父類(lèi),并傳遞參數(shù),需要的朋友可以參考下2025-03-03
Python中Timedelta轉(zhuǎn)換為Int或Float方式
這篇文章主要介紹了Python中Timedelta轉(zhuǎn)換為Int或Float方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07

