python中的socket套接字用于實現網絡通信
tcp套接字的使用流程如下:
- 創建套接字對象 --- clientSocket = socket(AF_INET,SOCK_STREAM)
- 連接服務器,指定服務器的ip地址和端口(參數是個元組)------------clientSocket.connect(('192.168.99.173',9999))
- 發送信息 -------------clientSocket.send(msg.encode('utf8'))
- 接收信息 ----------------msg = clientSocket.recv(1024)
我們先實現一個tcp的客戶端
由于socket在接收數據時默認為堵塞狀態,當接收數據是進程就刮起了,這時就無法發送信息,要實現全雙工的通信,需要給接收數據和發送數據各自分配一個線程。
from socket import *
from threading import Thread
def sendMsg(clientSocket):
while True:
msg = input('>>')
clientSocket.send(msg.encode('utf8'))
def recvMsg(clientSocket):
while True:
msg = clientSocket.recv(1024)
print('\r>>%s'%msg.decode('utf8'))
def main():
clientSocket = socket(AF_INET,SOCK_STREAM)
clientSocket.connect(('192.168.99.173',9999))
tr = Thread(target=recvMsg,args=(clientSocket,)) #將套接字作為參數傳給新線程,各自的線程中分別執行收,發數據
ts = Thread(target=sendMsg,args=(clientSocket,))
tr.start()
ts.start()
if __name__ == '__main__':
main()
接下來實現服務端
服務端的套接字比較特殊,因為是為客戶端服務,所以要確定下來ip地址和端口號以便客戶端訪問,然后服務器還要能生成新的客服套接字單獨為一個客戶端服務,利用多進程就可以生成多個同時工作的客服socket了。
服務器的套接字工作流程如下:
- 創建主套接字 --------serverSocket = socket(AF_INET,SOCK_STREAM)
- 綁定ip和端口 --------------------- serverSocket.bind(('',9999))
- 改為被動套接字 ------------------------------serverSocket.listen(5)
- 檢測客戶端連接 ,若連接則返回 新的服務套接字和客戶端地址ip----- newSocket,destAdr = serverSocket.accept()
- 每個服務套接字進行與客戶端的數據交換(在各自的進程里)
# 要求
#1. 使用tcp協議通信
#2. 使用多進程配合多線程配合多線程的方式實現多個全雙工的對話。
#實現
#1. 創建tcp的套接字,綁定,監聽,變為被動
#2. 每收到一次請求返回新的套接字,同時創建子進程,用來單獨用新套接字對話
#3. 子進程中創建兩個線程,用來收發數據
import os
from socket import *
from multiprocessing import Process
from threading import Thread,local
import time
def worker(newSocket,destAdr):
print('創建子會話進程成功..')
#local_school = local() #創建ThreadLocal對象,用來儲存各線程的局部變量
ts = Thread(target=sendMsg,args=(newSocket,))
tr = Thread(target=recvMsg,args=(newSocket,))
ts.start()
tr.start()
ts.join()
tr.join()
def sendMsg(newSocket):
print('發送進程準備完畢!')
print(os.getpid())
while True:
msg = str( os.getpid())
newSocket.send(msg.encode('utf8'))
time.sleep(5)
def ps_is_end():
print('子進程結束')
def recvMsg(newSocket):
print('接受進程準備完畢!')
while True:
msg = newSocket.recv(1024)
if msg.decode('utf8')!= '': #在ubantu中測試當客戶端連接關閉時會自動發送‘’,而在windows下運行卻會報錯
print('\r>>%s'%msg.decode('utf8'))
else:
print('%d下限了'%os.getpid())
newSocket.close()
break
def main():
serverSocket = socket(AF_INET,SOCK_STREAM)
serverSocket.bind(('',9999))
serverSocket.listen(5)
while True:
newSocket,destAdr = serverSocket.accept()
p = Process(target=worker,args=(newSocket,destAdr,))
p.start()
newSocket.close()#拷貝到了新的進程中,這里的可以刪掉了
serverSocket.close()
if __name__ == '__main__':
main()
試一下效果,先啟動服務器,再啟動客戶端,然后客戶端發條信息
捕獲.PNG
捕獲2.PNG