首先需要说明的一点是:这里并不会记录很深奥的socket编程,只是会分析一个最简单的socket编程聊天室下的几种特殊异常情况的处理,代码如下:

服务端:

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中按下调试窗口左端的stop按钮时),客户端当然会报错,但是在Linux系统上,服务端不会报错,只是认为接收到的信息为空,即此处的server_recv_data为空,所以在此服务端做了一个判断退出内层循环,以便可以连接其他的客户端发送来的连接请求;

2.当客户端输入exit表示要退出当前聊天的时候,此处客户端if判断break退出,关键是此时的服务端得到的结果是,server_recv_data为空,与上面情况一样;

3.唯一可以让服务端的server——recv_data = conn.recv(1024)发生阻塞的是:客户端直接按下回车,此时一个尴尬的情况是服务端仍然等待客户端发送信息(认为没有信息到来),客户端无法再发送信息,因为刚才已经按回车了,虽然可以再输入信息,但是无法发送了(即使再按回车也没用),所以为了解决这个问题,我们默认认为当客户端做出这个动作的时候表示要退出,此时inp为空,所以在客户端做出了if not inp:的判断,退出聊天,服务端收到为空,break之后继续等待其他连接