88

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

4

4 回答 4

65

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”,即使第二次关闭了套接字。

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

于 2008-09-30T07:41:15.173 回答
29

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

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

服务器:

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() 为真,所以我推荐这个
于 2011-11-25T11:33:01.773 回答
13

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

out.println(output);
if(out.checkError())
{
    throw new Exception("Error transmitting data.");
}
于 2009-10-29T22:52:46.810 回答
-12

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

于 2016-01-27T21:42:39.080 回答