web-dev-qa-db-ja.com

リモート側のソケットのクローズを検出する方法は?

Socket#close()がリモート側のソケットで呼び出されたかどうかをどのように検出しますか?

82
Kevin Wong

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

サーバーを起動し、クライアントを起動します。 2回目にソケットが閉じられても、「connected:true」が2回出力されることがわかります。

本当に見つける唯一の方法は、関連するInput/OutputStreamsを読み取る(戻り値として-1を取得する)か、書き込む(IOException(壊れたパイプ)がスローされる)ことです。

62
WMR

回答が異なるため、これをテストし、テスト例を含めて結果を投稿することにしました。

ここのサーバーはクライアントにデータを書き込むだけで、入力を期待しません。

サーバー:

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になるため、これをお勧めします
26

クライアントソケットへの書き込み中に、ソケット出力ストリームエラーをチェックすることもできます。

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