Python基于Socket實(shí)現(xiàn)簡(jiǎn)易多人聊天室的示例代碼
前言
套接字(Sockets)是雙向通信信道的端點(diǎn)。 套接字可以在一個(gè)進(jìn)程內(nèi),在同一機(jī)器上的進(jìn)程之間,或者在不同主機(jī)的進(jìn)程之間進(jìn)行通信,主機(jī)可以是任何一臺(tái)有連接互聯(lián)網(wǎng)的機(jī)器。
套接字可以通過(guò)多種不同的通道類型實(shí)現(xiàn):Unix域套接字,TCP,UDP等。 套接字庫(kù)提供了處理公共傳輸?shù)奶囟悾约耙粋€(gè)用于處理其余部分的通用接口。
socket模塊:
要?jiǎng)?chuàng)建套接字,必須使用套接字模塊中的socket.socket()函數(shù),該函數(shù)具有一般語(yǔ)法
s = socket.socket (socket_family, socket_type, protocol = 0)
| 參數(shù) | 描述 |
|---|---|
| socket_family | 它的值可以是:AF_UNIX或AF_INET,如前所述。 |
| socket_type | 它的值可以是:SOCK_STREAM或SOCK_DGRAM。 |
| protocol | 這通常被省略,默認(rèn)為0。 |
常用方法:
| 序號(hào) | 方法 | 描述 |
|---|---|---|
| 1 | s.bind() | 此方法將地址(主機(jī)名,端口號(hào)對(duì))綁定到套接字。 |
| 2 | s.recvfrom() | 此方法接收UDP消息,返回值是一對(duì)(字節(jié), 地址) ,其中字節(jié)是代表接收到的數(shù)據(jù)的字節(jié)對(duì)象,而地址是發(fā)送數(shù)據(jù)的套接字的地址 |
| 3 | s.sendto() | 此方法發(fā)送UDP消息,將數(shù)據(jù)發(fā)送到套接字。該套接字不應(yīng)連接到遠(yuǎn)程套接字,因?yàn)槟繕?biāo)套接字是由address指定的 |
| 4 | s.close() | 此方法關(guān)閉套接字,套接字對(duì)象上所有以后的操作都將失敗。遠(yuǎn)端將不再接收任何數(shù)據(jù)(在清除排隊(duì)的數(shù)據(jù)之后)。套接字在被垃圾回收時(shí)會(huì)自動(dòng)關(guān)閉 |
| 5 | socket.gethostname() | 返回主機(jī)名,返回一個(gè)字符串,其中包含當(dāng)前正在執(zhí)行Python解釋器的計(jì)算機(jī)的主機(jī)名。 |
示例1
服務(wù)器端
#sever.py
import socket
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
host = socket.gethostname()
port = 8088
s.bind((host,port))
try:
while True:
receive_data,addr = s.recvfrom(1024)
print("來(lái)自服務(wù)器" + str(addr) + "的消息:")
print(receive_data.decode('utf-8'))
msg = input('please input send to msg:')
s.sendto(msg.encode('utf-8'),addr)
except:
s.close()
客戶端
#client.py
import socket
s = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
try:
while True:
host = socket.gethostname()
port = 8088
send_data = input('please input msg:')
s.sendto(send_data.encode('utf-8'),(host,port))
msg,addr = s.recvfrom(1024)
print("來(lái)自服務(wù)器" + str(addr) + "的消息:")
print(msg.decode('utf-8'))
except:
s.close()
服務(wù)端示例

客戶端示例

簡(jiǎn)易的UDP聊天實(shí)現(xiàn)了,下面我們來(lái)優(yōu)化一下示例。
示例2
服務(wù)端:
#server.py
import socket
import logging
def main():
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # 創(chuàng)建socket對(duì)象
addr = ('127.0.0.1', 9999)
s.bind(addr) # 綁定地址和端口
logging.info('UDP Server on %s:%s...', addr[0], addr[1])
user = {} # 存放字典{addr:name}
while True:
try:
data, addr = s.recvfrom(1024) # 等待接收客戶端消息存放在2個(gè)變量data和addr里
if not addr in user: # 如果addr不在user字典里則執(zhí)行以下代碼
for address in user: # 從user遍歷數(shù)據(jù)出來(lái)address
s.sendto(data + ' 進(jìn)入聊天室...'.encode('utf-8'), address) # 發(fā)送user字典的data和address到客戶端
user[addr] = data.decode('utf-8') # 接收的消息解碼成utf-8并存在字典user里,鍵名定義為addr
continue # 如果addr在user字典里,跳過(guò)本次循環(huán)
if 'EXIT'.lower() in data.decode('utf-8'):#如果EXIT在發(fā)送的data里
name = user[addr] #user字典addr鍵對(duì)應(yīng)的值賦值給變量name
user.pop(addr) #刪除user里的addr
for address in user: #從user取出address
s.sendto((name + ' 離開了聊天室...').encode(), address) #發(fā)送name和address到客戶端
else:
print('"%s" from %s:%s' %(data.decode('utf-8'), addr[0], addr[1]))
for address in user: #從user遍歷出address
if address != addr: #address不等于addr時(shí)間執(zhí)行下面的代碼
s.sendto(data, address) #發(fā)送data和address到客戶端
except ConnectionResetError:
logging.warning('Someone left unexcept.')
if __name__ == '__main__':
main()
客戶端:
#clinet.py
import socket
import threading
def recv(sock, addr):
'''
一個(gè)UDP連接在接收消息前必須要讓系統(tǒng)知道所占端口
也就是需要send一次,否則win下會(huì)報(bào)錯(cuò)
'''
sock.sendto(name.encode('utf-8'), addr)
while True:
data = sock.recv(1024)
print(data.decode('utf-8'))
def send(sock, addr):
'''
發(fā)送數(shù)據(jù)的方法
參數(shù):
sock:定義一個(gè)實(shí)例化socket對(duì)象
server:傳遞的服務(wù)器IP和端口
'''
while True:
string = input('')
message = name + ' : ' + string
data = message.encode('utf-8')
sock.sendto(data, addr)
if string.lower() == 'EXIT'.lower():
break
def main():
'''
主函數(shù)執(zhí)行方法,通過(guò)多線程來(lái)實(shí)現(xiàn)多個(gè)客戶端之間的通信
'''
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
server = ('127.0.0.1', 9999)
tr = threading.Thread(target=recv, args=(s, server), daemon=True)
ts = threading.Thread(target=send, args=(s, server))
tr.start()
ts.start()
ts.join()
s.close()
if __name__ == '__main__':
print("-----歡迎來(lái)到聊天室,退出聊天室請(qǐng)輸入'EXIT(不分大小寫)'-----")
name = input('請(qǐng)輸入你的名稱:')
print('-----------------%s------------------' % name)
main()
支持多人的簡(jiǎn)易聊天室示例,多個(gè)客戶端通過(guò)一個(gè)服務(wù)器進(jìn)行之間通信

到此這篇關(guān)于Python基于Socket實(shí)現(xiàn)簡(jiǎn)易多人聊天室的示例代碼的文章就介紹到這了,更多相關(guān)Python Socket多人聊天室內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python實(shí)現(xiàn)根據(jù)主機(jī)名字獲得所有ip地址的方法
這篇文章主要介紹了python實(shí)現(xiàn)根據(jù)主機(jī)名字獲得所有ip地址的方法,涉及Python解析IP地址的相關(guān)技巧,需要的朋友可以參考下2015-06-06
python3個(gè)性簽名設(shè)計(jì)實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了python3個(gè)性簽名設(shè)計(jì)的實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06
使用Python實(shí)現(xiàn)毫秒級(jí)搶單功能
年中購(gòu)物618大狂歡開始了,各大電商又開始了大力度的折扣促銷,我們的小胖又給大家謀了一波福利,淘寶APP直接搜索:小胖發(fā)福利,每天領(lǐng)取三次粉絲專屬現(xiàn)金大紅包。這篇文章主要介紹了用Python完成毫秒級(jí)搶單,助你秒殺淘寶大單,需要的朋友可以參考下2019-06-06
pandas數(shù)值排序的實(shí)現(xiàn)實(shí)例
篩選和排序是Excel中使用頻率最多的功能,本文主要介紹了pandas數(shù)值排序的實(shí)現(xiàn)實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),,感興趣的可以了解一下2021-07-07
Python實(shí)現(xiàn)將字典內(nèi)容寫入json文件
這篇文章主要為大家詳細(xì)介紹了如何利用Python語(yǔ)言實(shí)現(xiàn)將字典內(nèi)容寫入json文件,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2022-08-08
詳解OpenCV執(zhí)行連通分量標(biāo)記的方法和分析
在本教程中,您將學(xué)習(xí)如何使用?OpenCV?執(zhí)行連通分量標(biāo)記和分析。具體來(lái)說(shuō),我們將重點(diǎn)介紹?OpenCV?最常用的連通分量標(biāo)記函數(shù):cv2.connectedComponentsWithStats,感興趣的可以了解一下2022-08-08
django admin 根據(jù)choice字段選擇的不同來(lái)顯示不同的頁(yè)面方式
這篇文章主要介紹了django admin 根據(jù)choice字段選擇的不同來(lái)顯示不同的頁(yè)面方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-05-05

