java - 检查 Java TCP 服务器上的客户端断开连接 - 仅输出

标签 java sockets tcp

我有一个 Java TCP 服务器,当客户端连接到它时,它会每 30 秒向客户端输出一条消息。严格要求客户端不向服务器发送任何消息,并且服务器不向客户端发送除30秒间隔消息之外的任何数据。

当我断开客户端连接时,服务器直到下次尝试写入客户端时才会意识到这一点。因此,服务器最多可能需要 30 秒才能识别断开连接。

我想做的是每隔几秒检查一次断开连接而不必等待,但我不确定如何做到这一点,因为 a) 服务器没有从客户端接收 b) 服务器不能发送任何其他数据。任何人都可以对此有所了解吗?谢谢。

最佳答案

即使您的服务器没有从客户端“接收”,客户端套接字上的非阻塞读取也会告诉您没有可读取的内容(正如您所期望的),或者客户端已断开连接。

如果您正在使用 NIO,您可以简单地使用非阻塞 Selector 循环(使用非阻塞套接字)并且只在 30 秒标记上写入。如果 SelectionKey 是可读的并且对 SocketChannel 的读取返回 -1,则您知道客户端已断开连接。

编辑:另一种阻塞方法是简单地选择 30 秒超时。任何客户端断开连接都会导致选择返回,您将通过读取集知道哪些是那些。您需要做的另一件事是跟踪您在选择中被阻止的时间,以确定何时在 30 秒标记上进行写入(将下一次选择的超时设置为增量)。

大修改:在与下面的 Myn 交谈后,提供了完整的示例:

public static void main(String[] args) throws IOException {

    ServerSocket serverSocket = null;
    try {
        serverSocket = new ServerSocket(4444);
    } catch (IOException e) {
        System.err.println("Could not listen on port: 4444.");
        System.exit(1);
    }

    Socket clientSocket = null;
    try {
        clientSocket = serverSocket.accept();
    } catch (IOException e) {
        System.err.println("Accept failed.");
        System.exit(1);
    }

    // Set a 1 second timeout on the socket
    clientSocket.setSoTimeout(1000);

    PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true);
    BufferedReader in = new BufferedReader(
            new InputStreamReader(
            clientSocket.getInputStream()));

    long myNextOutputTime = System.currentTimeMillis() + 30000;

    String inputLine = null;
    boolean connected = true;
    while (connected)
    {
        try {
            inputLine = in.readLine();
            if (inputLine == null)
            {
                System.out.println("Client Disconnected!");
                connected = false;
            }
        }
        catch(java.net.SocketTimeoutException e)
        {
            System.out.println("Timed out trying to read from socket");
        }

        if (connected && (System.currentTimeMillis() - myNextOutputTime > 0))
        {
            out.println("My Message to the client");
            myNextOutputTime += 30000;
        }


    }

    out.close();
    in.close();
    clientSocket.close();
    serverSocket.close();
}

这里值得注意的是 PrintWriter 确实使您远离实际套接字,并且您不会在写入时捕获套接字断开连接(它永远不会抛出异常,你必须使用 checkError() 手动检查它)您可以改为使用 BufferedWriter(需要使用 flush() 来推送输出)并像 BufferedReader 一样处理它以在写入时捕捉迪斯科。

关于java - 检查 Java TCP 服务器上的客户端断开连接 - 仅输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5263565/

相关文章:

c - UDP 服务器只响应一个客户端

c++ - 多个应用程序如何使用同一个网络端口?

ruby HttpClient 库在响应持久连接后关闭套接字?

java - Nodejs 在 Java 程序成功 POST 时并不总是触发 'data' 事件

c - 使用 C 和 UDP 实现停止并等待,什么导致 'resource temporarily unavailable'?

java - 如何使用php从Android将数据插入mysql数据库?

c - 绑定(bind) : Address Already in Use

sockets - 两个应用程序可以监听同一个端口吗?

java - Google 应用引擎部署错误 : JDK not JRE

java - 动 Canvas 局 JavaFX