首先需要说明的一点是:这里并不会记录很深奥的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之后继续等待其他连接