【问题标题】:How to detect a remote side socket close? [duplicate]如何检测远程侧套接字关闭? [复制]
【发布时间】:2010-09-14 04:11:57
【问题描述】:

如何检测Socket#close() 是否已在远程端的套接字上调用?

【问题讨论】:

    标签: java networking sockets tcp


    【解决方案1】:

    isConnected 方法无济于事,即使远程端关闭了套接字,它也会返回true。试试这个:

    public class MyServer {
        public static final int PORT = 12345;
        public static void main(String[] args) throws IOException, InterruptedException {
            ServerSocket ss = ServerSocketFactory.getDefault().createServerSocket(PORT);
            Socket s = ss.accept();
            Thread.sleep(5000);
            ss.close();
            s.close();
        }
    }
    
    public class MyClient {
        public static void main(String[] args) throws IOException, InterruptedException {
            Socket s = SocketFactory.getDefault().createSocket("localhost", MyServer.PORT);
            System.out.println(" connected: " + s.isConnected());
            Thread.sleep(10000);
            System.out.println(" connected: " + s.isConnected());
        }
    }
    

    启动服务器,启动客户端。您会看到它打印了两次“connected: true”,即使第二次关闭了套接字。

    真正找出答案的唯一方法是在关联的 Input/OutputStreams 上读取(您将得到 -1 作为返回值)或写入(将抛出 IOException(损坏的管道))。

    【讨论】:

    • 在您的示例中,服务器没有正常关闭 TCP 连接(发送了 RST,仅此而已)。如果您在客户端套接字上调用 close(),则连接将正常关闭。您可以通过读取操作的不同结果看到这一点:抛出 SocketException 与读取返回 -1。
    • 顺便说一句,感谢您提醒我 Java 套接字工作方式中的这个“奥秘”。结果我问了一个后续问题:stackoverflow.com/questions/155243/…
    • 找到答案的唯一方法就是写。从突然关闭的套接字读取可能并不总是返回 -1。 stackoverflow.com/a/6404085/372643
    • @Pacerier,请参阅我发布的链接。本质上,一个不发送任何东西的连接和一个死掉的连接之间没有区别(如果你没有使用 TCP keep-alive)。您只能通过写入来检测已关闭的连接。
    • 为什么会有这么多赞成票:/ 它真的连问题都没有回答,它只是在解释什么不起作用...
    【解决方案2】:

    由于答案有偏差,我决定对此进行测试并发布结果 - 包括测试示例。

    这里的服务器只是将数据写入客户端,并不期望任何输入。

    服务器:

    ServerSocket serverSocket = new ServerSocket(4444);
    Socket clientSocket = serverSocket.accept();
    PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true);
    while (true) {
      out.println("output");
      if (out.checkError()) System.out.println("ERROR writing data to socket !!!");
      System.out.println(clientSocket.isConnected());
      System.out.println(clientSocket.getInputStream().read());
            // thread sleep ...
      // break condition , close sockets and the like ...
    }
    
    • clientSocket.isConnected() 一旦客户端连接(甚至在断开连接之后)总是返回真值!!
    • getInputStream().read()
      • 只要客户端已连接,线程就会等待输入,因此您的程序不会执行任何操作 - 除非您收到一些输入
      • 如果客户端断开连接,则返回 -1
    • 客户端断开连接后,out.checkError() 为真,所以我推荐这个

    【讨论】:

    • checkError() 在写入时一旦检测到错误就为真,并且不会更早。一旦客户端断开连接,它肯定不会成为真的。 -1
    【解决方案3】:

    您还可以在写入客户端套接字时检查套接字输出流错误。

    out.println(output);
    if(out.checkError())
    {
        throw new Exception("Error transmitting data.");
    }
    

    【讨论】:

    • 可以,但不一定会立即获得。
    【解决方案4】:

    如果远程系统断开/关闭连接,Socket.Available 方法将立即抛出 SocketException。

    【讨论】:

    • 不,不会。错误的。错误的。错误信息。
    猜你喜欢
    • 2014-10-21
    • 2015-12-09
    • 2020-05-12
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-08-20
    相关资源
    最近更新 更多