这个问题在这里已经有答案了:

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

有帮助吗?

解决方案

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());
    }
}

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

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

其他提示

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

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

服务器:

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() 总是返回 true ,这很奇怪!
  • getInputStream().read()
    • 只要客户端连接,线程就等待输入,因此使您的程序不执行任何操作 - 除非您获得一些输入
    • 如果客户端断开连接,则返回 -1
  • 一旦客户端断开连接,out.checkError() 就会为 true,所以我推荐这个

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

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

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

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top