首先需要说明的一点是:这里并不会记录很深奥的socket编程,只是会分析一个最简单的socket编程聊天室下的几种特殊异常情况的处理,代码如下:
服务端:
1 import socket 2 3 HOST = "" 4 PORT = 8870 5 sk = socket.socket(socket.AF_INET, socket.SOCK_STREAM) 6 sk.bind((HOST, PORT)) 7 sk.listen(5) 8 while True: 9 print("服务器启动") 10 conn, addr = sk.accept() 11 print(addr) 12 while True: 13 try: 14 server_recv_data = conn.recv(1024) #若客户端输入exit退出,则此处并没有阻塞,只是收到的信息为空 15 #若客户端输入空,即直接按下回车键,那么此处将会被阻塞 16 if len(server_recv_data) == 0: 17 print("收到为空") 18 break 19 except ConnectionResetError as err: 20 print(err) 21 break 22 print(str(server_recv_data, encoding="utf-8")) 23 server_resp_data = input(">>>") 24 conn.sendall(bytes(server_resp_data, encoding="utf-8")) 25 conn.close() 26 sk.close()
客户端:
1 import socket 2 3 HOST = "127.0.0.1" 4 PORT = 8870 5 sk = socket.socket() 6 sk.connect((HOST, PORT)) 7 print("客户端启动...") 8 9 while True: 10 inp = input(">>>") 11 if inp == "exit"or not inp: #此处需要说明的是,当直接在客户端按下回车enter键的时候,inp为空 12 break 13 sk.sendall(bytes(inp, encoding="utf-8")) 14 server_response = sk.recv(1024) 15 print(str(server_response, encoding="utf-8")) 16 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之后继续等待其他连接
时间: 2024-12-24 13:09:28