用socket和多進程配合多線程實現簡單的tcp的聊天服務器

python中的socket套接字用于實現網絡通信
tcp套接字的使用流程如下:

  1. 創建套接字對象 --- clientSocket = socket(AF_INET,SOCK_STREAM)
  2. 連接服務器,指定服務器的ip地址和端口(參數是個元組)------------clientSocket.connect(('192.168.99.173',9999))
  3. 發送信息 -------------clientSocket.send(msg.encode('utf8'))
  4. 接收信息 ----------------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了。
服務器的套接字工作流程如下:

  1. 創建主套接字 --------serverSocket = socket(AF_INET,SOCK_STREAM)
  2. 綁定ip和端口 --------------------- serverSocket.bind(('',9999))
  3. 改為被動套接字 ------------------------------serverSocket.listen(5)
  4. 檢測客戶端連接 ,若連接則返回 新的服務套接字和客戶端地址ip----- newSocket,destAdr = serverSocket.accept()
  5. 每個服務套接字進行與客戶端的數據交換(在各自的進程里)
# 要求
#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
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • 一: 網絡各個協議:TCP/IP、SOCKET、HTTP 網絡七層由下往上分別為物理層、數據鏈路層、網絡層、傳輸層...
    iYeso閱讀 1,453評論 0 13
  • 網絡由下往上分為物理層、數據鏈路層、網絡層、傳輸層、會話層、表示層和應用層。通過初步的了解,我知道IP協議對應于網...
    黑馬飲清泉閱讀 1,501評論 0 6
  • 5 置房 解放軍來了,土匪胡子被消滅了,村子恢復了往日的寧靜。 三爺這些年也攢了不少積蓄,解放后,經人介紹娶了楊家...
    肅山閱讀 303評論 0 2
  • 《媽媽團團zhuan》? zhuan??? 賺?轉?撰? 賺的是白花花的銀子和和諧的家庭關系! 轉的是對生活的態度...
    指甲月閱讀 134評論 0 0
  • 文案:更漏短,歲月長。你看這個市井陌上,天下之處,哪朝那代哪個人不是為自已而活著! 是夜,今晚月色并不是太好。那...
    千山4499閱讀 354評論 1 0