首先需要說明的一點是:這裡並不會記錄很深奧的socket編程,只是會分析一個最簡單的socket編程聊天室下的幾種特殊異常情況的處理,代碼如下:
服務端:
import socketHOST = ""PORT = 8870sk = socket.socket(socket.AF_INET, socket.SOCK_STREAM)sk.bind((HOST, PORT))sk.listen(5)while True: print("伺服器啟動") conn, addr = sk.accept() print(addr) while True: try: server_recv_data = conn.recv(1024) #若用戶端輸入exit退出,則此處並沒有阻塞,只是收到的資訊為空白 #若用戶端輸入空,即直接按下斷行符號鍵,那麼此處將會被阻塞 if len(server_recv_data) == 0: print("收到為空白") break except ConnectionResetError as err: print(err) break print(str(server_recv_data, encoding="utf-8")) server_resp_data = input(">>>") conn.sendall(bytes(server_resp_data, encoding="utf-8")) conn.close()sk.close()
用戶端:
import socketHOST = "127.0.0.1"PORT = 8870sk = socket.socket()sk.connect((HOST, PORT))print("用戶端啟動...")while True: inp = input(">>>") if inp == "exit"or not inp: #此處需要說明的是,當直接在用戶端按下斷行符號enter鍵的時候,inp為空白 break sk.sendall(bytes(inp, encoding="utf-8")) server_response = sk.recv(1024) print(str(server_response, encoding="utf-8"))sk.close()
注意:
1.當用戶端強行關閉的時候(比如在pycharm中按下調試視窗左端的stop按鈕時),用戶端當然會報錯,但是在Linux系統上,服務端不會報錯,只是認為接收到的資訊為空白,即此處的server_recv_data為空白,所以在此服務端做了一個判斷退出內層迴圈,以便可以串連其他的用戶端發送來的串連請求;
2.當用戶端輸入exit表示要退出當前聊天的時候,此處用戶端if判斷break退出,關鍵是此時的服務端得到的結果是,server_recv_data為空白,與上面情況一樣;
3.唯一可以讓服務端的server——recv_data = conn.recv(1024)發生阻塞的是:用戶端直接按下斷行符號,此時一個尷尬的情況是服務端仍然等待用戶端發送資訊(認為沒有資訊到來),用戶端無法再發送資訊,因為剛才已經按斷行符號了,雖然可以再輸入資訊,但是無法發送了(即使再按斷行符號也沒用),所以為瞭解決這個問題,我們預設認為當用戶端做出這個動作的時候表示要退出,此時inp為空白,所以在用戶端做出了if not inp:的判斷,退出聊天,服務端收到為空白,break之後繼續等待其他串連