Python 使用socket實現一對多通信


這個折磨了我快一天的時間,查看官網的socket入門例子,都是一對一的,服務器是處理一對一的形式。如果讓服務器處理多個客戶端,使用Python提供的socketserver函數和select也是可以解決的,但是這兩個只能處理客戶端發過來的信息,不能自動發起向已經連接到服務器的某個客戶端進行通信,經過了多方的查找資料和總結,終於使用socket和threading解決了這個問題。

服務器代碼:

  

 1 from tkinter import *
 2 from socket import *
 3 import threading
 4 
 5 
 6 address='0.0.0.0'
 7 port=9000
 8 buffsize=1024
 9 s = socket(AF_INET, SOCK_STREAM)
10 s.bind((address,port))
11 s.listen(5)     #最大連接數
12 conn_list = []
13 conn_dt = {}
14 
15 def tcplink(sock,addr):
16     while True:
17         try:
18             recvdata=sock.recv(buffsize).decode('utf-8')
19             print(recvdata, addr)
20             gui.infoList.config(state=NORMAL)
21             gui.infoList.insert(END, addr, 'name')
22             gui.infoList.insert(END, ':\t')
23             gui.infoList.insert(END, recvdata, 'conment')
24             gui.infoList.insert(END, '\n\n')
25             gui.infoList.config(state=DISABLED)
26             if not recvdata:
27                 break
28         except:
29             sock.close()
30             print(addr,'offline')
31             _index = conn_list.index(addr)
32             gui.listBox.delete(_index)
33             conn_dt.pop(addr)
34             conn_list.pop(_index)
35             break
36 
37 def recs():
38     while True:
39         clientsock,clientaddress=s.accept()
40         if clientaddress not in conn_list:
41             conn_list.append(clientaddress)
42             conn_dt[clientaddress] = clientsock
43             gui.listBox.insert(END, clientaddress)
44         print('connect from:',clientaddress)
45         #在這里創建線程,就可以每次都將socket進行保持
46         t=threading.Thread(target=tcplink,args=(clientsock,clientaddress))
47         t.start()
48 
49 
50 class GUI:
51     def __init__(self, root):
52         self.root = root
53         self.leftFrame = Frame(self.root, width=20, height=30)
54         self.leftFrame.grid(row=0, column=0)
55         self.rightFrame = Frame(self.root, width=20, height=30)
56         self.rightFrame.grid(row=0, column=1)
57         Label(self.leftFrame, text='在線IP地址列表').grid(row=0, column=0)
58         self.listBox = Listbox(self.leftFrame, width=15, height=10)
59         self.listBox.grid(row=1, column=0)
60         self.entry = Entry(self.rightFrame, font=('Serief', 18), width=30)
61         self.entry.grid(row=0, column=0)
62         self.sendBtn = Button(self.rightFrame, text='發送', command=self.send, width=10)
63         self.sendBtn.grid(row=0, column=1)
64         Label(self.rightFrame, text='聊天信息').grid(row=1, columnspan=2)
65         self.infoList = Text(self.rightFrame, width=40, height=12)
66         self.infoList.grid(row=2, columnspan=2)
67         self.infoList.tag_config('name', background='yellow', foreground='red')
68         self.infoList.tag_config('conment', background='black', foreground='white')
69 
70 
71     def send(self):
72         _index = self.listBox.curselection()
73         conn_dt[self.listBox.get(_index)].sendall(self.entry.get().encode('utf-8'))
74         self.entry.delete(0, END)
75 
76 def createGUI():
77     global gui
78     root = Tk()
79     gui = GUI(root)
80     root.title('服務器')
81     root.mainloop()
82 
83 if __name__ == '__main__':
84     t1 = threading.Thread(target=recs, args=(), name='rec')
85     t2 = threading.Thread(target=createGUI, args=(), name='GUI')
86 
87     t1.start()
88     t2.start()

客戶端代碼:

  

 1 from socket import *
 2 import threading
 3 from tkinter import *
 4 
 5 address='127.0.0.1'   #服務器的ip地址
 6 port=9000
 7 buffsize=1024
 8 s=socket(AF_INET, SOCK_STREAM)
 9 s.connect((address,port))
10 
11 
12 
13 def recv():
14     while True:
15         recvdata = s.recv(buffsize).decode('utf-8')
16         gui.listBox.insert(END, recvdata)
17         print('\n' + recvdata + '\n')
18 
19 class GUI:
20     def __init__(self, root):
21         self.root = root
22         self.listBox = Listbox(self.root)
23         self.listBox.pack()
24         self.entry = Entry(self.root)
25         self.entry.pack()
26         self.sendBtn = Button(self.root, text='發送', command=self.send)
27         self.sendBtn.pack()
28 
29     def send(self):
30         senddata = self.entry.get()
31         s.send(senddata.encode())
32 
33 def createGUI():
34     global gui
35     root = Tk()
36     gui = GUI(root)
37     root.title('客戶端')
38     root.mainloop()
39 
40 if __name__ == '__main__':
41     t1 = threading.Thread(target=recv, args=(), name='recv')
42     t2 = threading.Thread(target=createGUI, args=(), name='gui')
43 
44     t1.start()
45     t2.start()

加上了界面,可以實現服務端與已連接的某個客戶端進行通信,可以處理多個客戶端的通信,


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM