简体   繁体   English

Python套接字如何处理客户端意外关闭

[英]Python sockets how to handle the client closing unexpectedly

My python socket server listens and then connects to a client that will then send a indeterminate number of strings from a user. 我的python套接字服务器侦听,然后连接到客户端,该客户端将随后从用户发送不确定数量的字符串。 The client may then close or lose connection to the server. 然后,客户端可能会关闭或失去与服务器的连接。

This causes an error. 这会导致错误。

[Errno 10053] An established connection was aborted by the software in your host machine. [Errno 10053]主机中的软件中断了已建立的连接。 or [Errno 10054] An existing connection was forcibly closed by the remote host 或[Errno 10054]现有连接被远程主机强行关闭

How do I handle this event so that I can close the connection and restart my server listening for a reconnect? 如何处理此事件,以便可以关闭连接并重新启动服务器以侦听重新连接?

Python Server example: Python Server示例:

# Echo server program
import socket
import sys

HOST = None               # Symbolic name meaning all available interfaces
PORT = 7001              # Arbitrary non-privileged port
s = None

def startListening():
    print "starting to listen"

    for res in socket.getaddrinfo(HOST, PORT, socket.AF_UNSPEC,
                                  socket.SOCK_STREAM, 0, socket.AI_PASSIVE):
        af, socktype, proto, canonname, sa = res
        try:
            s = socket.socket(af, socktype, proto)
        except socket.error as msg:
            s = None
            continue
        try:
            s.bind(sa)
            s.listen(1)
        except socket.error as msg:
            s.close()
            s = None
            continue
        break
    if s is None:
        print 'could not open socket'
        sys.exit(1)
    conn, addr = s.accept()
    print 'Connected by', addr
    while 1:
        data = conn.recv(1024)
        if not data: 
            break
        print data
        message = ""
        while not "quit" in message:
            message = raw_input('Say Something : ')
            conn.sendall(message)
        #conn.send("I got that, over!")

    conn.close()
    print "connection closed"

while 1:
    startListening()

python client example: python客户端示例:

# Echo client program
import socket
import sys

HOST = 'localhost'    # The remote host
PORT = 7001              # The same port as used by the server
s = None
for res in socket.getaddrinfo(HOST, PORT, socket.AF_UNSPEC, socket.SOCK_STREAM):
    af, socktype, proto, canonname, sa = res
    try:
        s = socket.socket(af, socktype, proto)
    except socket.error as msg:
        s = None
        continue
    try:
        s.connect(sa)
    except socket.error as msg:
        s.close()
        s = None
        continue
    break
if s is None:
    print 'could not open socket'
    sys.exit(1)
s.sendall("Please send me some strings")
data = ""
while ("quit" not in data):
    data = s.recv(1024)
    print 'Received', repr(data)
s.close()

To reproduce this error, run the server in one command window and the client in a second, then close the client window. 要重现此错误,请在一个命令窗口中运行服务器,然后在第二个命令窗口中运行客户端,然后关闭客户端窗口。

Adding try: except: around the data send and data recv in the server script appears to mitigate the problem: 添加try:例外:服务器脚本中的数据发送和数据接收周围出现,以缓解该问题:

# Echo server program
import socket
import sys

HOST = None               # Symbolic name meaning all available interfaces
PORT = 7001              # Arbitrary non-privileged port
s = None

def startListening():
    print "starting to listen"

    for res in socket.getaddrinfo(HOST, PORT, socket.AF_UNSPEC,
                                  socket.SOCK_STREAM, 0, socket.AI_PASSIVE):
        af, socktype, proto, canonname, sa = res
        try:
            s = socket.socket(af, socktype, proto)
        except socket.error as msg:
            s = None
            continue
        try:
            s.bind(sa)
            s.listen(1)
        except socket.error as msg:
            s.close()
            s = None
            continue
        break
    if s is None:
        print 'could not open socket'
        sys.exit(1)
    conn, addr = s.accept()
    print 'Connected by', addr
    while 1:
        try: 
            data = conn.recv(1024)
        except:
            print "cannot recieve data"
            break
        if not data: 
            break
        print data
        message = ""
        while not "quit" in message:
            message = raw_input('Say Something : ')
            try: 
                conn.sendall(message)
            except Exception as exc:
                #print exc # or something.
                print "message could not be sent"
                break
        #conn.send("I got that, over!")

    conn.close()
    print "connection closed"

while 1:
    startListening()

Simply close your side of the connection and be ready to accept a new connection again. 只需关闭连接侧并准备好再次接受新连接即可。

The best way to handle this is to separate the creation of the server socket and the accept/read/write loop: 解决此问题的最佳方法是将服务器套接字的创建和accept / read / write循环分开:

create_server_socket()
while True:
    accept_new_connection();
    try:
        read_request();
        write_response()
    finally:
        close_connection()

我认为,当客户端意外关闭时,根据SIGPIPE,您应该处理信号

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM