当前位置: 首页 > 面试题库 >

检查ClientSocket在Java中是否已断开连接

傅献
2023-03-14
问题内容

基本上,我有一个服务器循环,用于管理与一个单独客户端的连接。在循环的某一时刻,如果存在ClientSocket,它将尝试读取以检查客户端是否仍然连接:

if (bufferedReader.read()==-1 ) {
       logger.info("CONNECTION TERMINATED!");
       clientSocket.close(); 
       setUpSocket(); //sets up the server to reconnect to the client
}else{
        sendHeartBeat(); //Send a heartbeat to the client
}

问题是,一旦创建了套接字,应用程序将挂起读取,我假设正在等待永远不会到来的数据,因为客户端永远不会发送到服务器。在此之前还可以,因为正确处理了断开连接(客户端断开连接后读取将最终失败),并且循环将尝试重新建立连接。但是,我现在添加了上面的sendHeartBeat()方法,该方法会定期让客户端知道服务器仍在运行。如果读取保留线程,则心跳永远不会发生!

因此,我假设我正在测试连接是否仍不正确。作为一个快速的技巧,我可以在一个单独的线程中运行bufferedReader.read(),但是那样的话,我会遇到各种各样我实际上不想处理的并发问题。

因此,问题有几方面:1)我是否检查客户端是否正确断开连接?2)如果没有,该怎么办?3)如果我做得正确,如何使读取不被进程扣为人质?还是穿线是唯一的方法?


问题答案:

创建套接字时,请先设置超时:

private int timeout    = 10000;
private int maxTimeout = 25000;

clientSocket.setSoTimeout(timeout);

这样,如果读取超时,您将获得java.net.SocketTimeoutException(必须抓住)。因此,您可以执行以下操作,假设您之前已如上所示设置了SO_TIMEOUT,并假定心跳将始终从远程系统获得响应:

volatile long lastReadTime;

try {
    bufferedReader.read();
    lastReadTime = System.currentTimeMillis();
} catch (SocketTimeoutException e) {
    if (!isConnectionAlive()) {
        logger.info("CONNECTION TERMINATED!");
        clientSocket.close(); 
        setUpSocket(); //sets up the server to reconnect to the client
    } else {
        sendHeartBeat(); //Send a heartbeat to the client
    }
}

public boolean isConnectionAlive() {
    return System.currentTimeMillis() - lastReadTime < maxTimeout;
}

解决此问题的一种常见方法是将超时设置为某个数字(例如10秒),然后跟踪上一次成功从套接字读取的时间。如果您的超时时间已达到2.5倍,请放弃客户端并关闭套接字(以防万一,将FIN数据包发送到另一端)。

如果心跳信号 _不会_从远程系统获得任何响应,而只是在连接断开时最终最终更早地产生IOException的一种方式,那么您可以这样做(假设sendHeartBeat本身不会抛出IOException):

try {
    if (bufferedReader.read() == -1) {
        logger.info("CONNECTION TERMINATED with EOF!");
        resetConnection();
    }
} catch (SocketTimeoutException e) {
    // This just means our read timed out ... the socket is still good
    sendHeartBeat(); //Send a heartbeat to the client
} catch (IOException e) {
    logger.info("CONNECTION TERMINATED with Exception " + e.getMessage());
    rehtml" target="_blank">setConnection();
}

....

private void resetConnection() {
    clientSocket.close(); 
    setUpSocket(); //sets up the server to reconnect to the client
}


 类似资料:
  • 我正在用java创建一个jframe。但我想检查系统上是否已经存在jframe。我尝试过一种方法,从任务管理器获取所有进程,然后查看它是否正在运行,但它不会工作。我使用了这篇文章中的一种方法:https://stackoverflow.com/a/19005828/15349408它的进程名是“javaw.exe”。 它无法检查jframe对象是否为null,因为它将从其他程序运行。是否可以检查j

  • 问题内容: 我需要编写一个自定义的批处理文件重命名器。我已经完成了大部分工作,除了我不知道如何检查文件是否已经打开。我只是在使用软件包,并且有一种方法,但是似乎无法测试该文件是否正在被另一个程序使用。关于如何进行这项工作的任何想法? 问题答案: 使用Apache Commons IO库…

  • 我想检查一下我的客户是否断开了连接。根据我的研究,一种可能的方式是我可以继续给客户写信。如果客户端未能接收到消息,则表示已断开连接。我向相应的客户端发送消息“Checking Connection:Client”clientNo。 我将clientNo 1连接到服务器,它接收 客户1 但是当我将另一个客户端连接到服务器时,我的clientNo1停止接收来自服务器的消息,我的clientNo2现在将

  • 问题内容: 你如何检查是否可以通过Java连接到Internet?一种方法是: 但是是否有更合适的方法来执行该任务,特别是如果你需要经常进行连续检查并且很有可能断开互联网连接时? 问题答案: 你应该连接到实际应用程序需要的地方。否则,你要测试是否与某个无关的地方(在这种情况下为Google)建立了连接。 特别是,如果你要尝试与Web服务进行通信,并且要控制该Web服务,则最好使用某种廉价的“获取状

  • 问题内容: 我正在Linux中使用termios api与串行设备进行通信。我正在尝试检测设备是否已断开连接,因此我可以尝试在超时后重新连接。我有以下示例代码: 在ttyUSB设备物理断开连接后,我实际上没有看到select()或fcntl返回错误值(-1)。我当然可以检查/ dev /中的文件是否存在,但是我希望有一个更优雅的解决方案。 希望得到任何建议,谢谢! 问题答案: 首先值得一提的是,s

  • 问题内容: 如何检查连接是否已在事务中?我正在使用Microsoft SQL Server数据库文件。 问题答案: 经过一番搜索,我发现了另一个“堆栈溢出”问题。事实证明,您不能在ADO.NET中嵌套事务。尝试时,您可能最终会启动两个不相关的事务,这会导致并行事务错误。 要查看连接当前是否在事务中,可以执行以下操作: 这将返回嵌套事务的数量。 请注意,您可以手动嵌套事务,而无需使用SqlTrans