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