最初に説明する必要があるのは、この記事では、最も単純なソケット プログラミング チャット ルームでのいくつかの特殊な例外の処理を分析するだけであるということです。
サーバー:
。import socket HOST = "" PORT = 8870 sk = 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 socket HOST = "127.0.0.1" PORT = 8870 sk = 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 のデバッグ ウィンドウの左側にある停止ボタンを押した場合など)、クライアントはもちろんエラーを報告しますが、 Linuxシステム、サーバー エラーは報告されませんが、受信した情報は空、つまりここでのserver_recv_dataは空であるとみなされます。そのため、サーバーは接続に接続できるように内部ループを終了する判断を行います。他のクライアントから送信されたリクエスト
2. クライアントが現在のチャットを終了するために exit に入ると、ここでクライアントは終了するために Break を判断します。この時点でサーバーによって取得された結果は、server_recv_data が空であるということです。上記の状況と同じです。
3. このとき、server-recv_data = conn.recv(1024) がブロックされる可能性があるのは、サーバーが直接 Enter キーを押した場合だけです。クライアントが情報を送信するのをまだ待機しています (情報が到着していないと考えられます)。クライアントは、今 Enter キーを押したため、情報を送信できません。情報を再度入力することはできますが、(もう一度 Enter キーを押しても) 送信することはできません。したがって、この問題を解決するために、クライアントがこのアクションを実行すると、この時点で inp が空であるため、クライアントは inp でない場合に判断します。サーバーは空のメッセージを受信し、休憩後も他の接続を待ち続けます