如何检测远程侧套接字关闭? [复制]

Posted

技术标签:

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

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

【问题讨论】:

【参考方案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 套接字工作方式中的这个“奥秘”。结果我问了一个后续问题:***.com/questions/155243/… 找到答案的唯一方法就是写。从突然关闭的套接字读取可能并不总是返回 -1。 ***.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。

【讨论】:

不,不会。错误的。错误的。错误信息。

以上是关于如何检测远程侧套接字关闭? [复制]的主要内容,如果未能解决你的问题,请参考以下文章

检测非阻塞套接字上的关闭连接

Ruby在recv时检测套接字关闭

远程协助开发总结

如何检测套接字连接何时丢失?

如何检测 Windows 中打开的套接字?

TCP套接字:检测对等方是不是在发送前关闭? (Linux)