检查Java TCP服务器上的客户端断开连接-仅输出
我有一个Java TCP服务器,当客户端连接到它时,它每30秒向客户端输出一条消息。严格要求客户端不向服务器发送任何消息,并且服务器不向客户端发送30秒间隔消息以外的任何数据 当我断开客户端连接时,服务器在下次尝试写入客户端之前不会意识到这一点。因此,服务器识别断开连接可能需要30秒检查Java TCP服务器上的客户端断开连接-仅输出,java,sockets,tcp,Java,Sockets,Tcp,我有一个Java TCP服务器,当客户端连接到它时,它每30秒向客户端输出一条消息。严格要求客户端不向服务器发送任何消息,并且服务器不向客户端发送30秒间隔消息以外的任何数据 当我断开客户端连接时,服务器在下次尝试写入客户端之前不会意识到这一点。因此,服务器识别断开连接可能需要30秒 我想做的是每隔几秒钟检查一次断开连接,而不必等待,但我不确定如何执行此操作,因为a)服务器没有从客户端接收到数据,b)服务器无法发送任何其他数据。有人能解释一下吗?谢谢。即使您的服务器没有从客户端“接收”,客户端套
我想做的是每隔几秒钟检查一次断开连接,而不必等待,但我不确定如何执行此操作,因为a)服务器没有从客户端接收到数据,b)服务器无法发送任何其他数据。有人能解释一下吗?谢谢。即使您的服务器没有从客户端“接收”,客户端套接字上的非阻塞读取也会告诉您没有要读取的内容(如您所料),或者客户端已断开连接 如果您使用的是NIO,您只需使用非阻塞
选择器循环(带有非阻塞套接字),只需在30秒的标记上写入即可。如果SelectionKey
是可读的,并且SocketChannel
上的读取返回-1,则您知道客户端已断开连接
编辑:另一种阻塞方法是只需选择30秒超时。任何客户端断开连接都会导致select返回,您将通过读取集知道哪些是select。您需要做的另一件事是跟踪您在选择中被阻止的时间,以确定何时在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()
)手动检查)。您可以改为使用缓冲写入程序(需要使用<代码> FrHuSH)/代码>来推导出输出,并像<代码> BufferedReader <代码>那样处理它,以在写上捕获一个DISCO。 < P>如果您管理多个客户端,那么我猜您将使用非阻塞套接字(如果没有,则考虑使用非阻塞)。。您可以使用选择器监视所有连接的套接字,以检查它们是否可读或可写,或者该套接字上是否存在错误。当某些客户端断开连接时,选择器将标记该套接字并返回。
要获得更多帮助,请使用google“Socket Select function”+1,或者换句话说,您必须使用非阻塞读取进行轮询,比如说每秒轮询一次。谢谢Brian。我没有使用NIO,还可以使用非阻塞读取吗?我也尝试过使用setSoTimeout()
但当我强制关闭客户端时,这似乎不起作用,这是我的测试之一。你不使用NIO有什么原因吗?它从2002年就开始使用了,这让你的问题变得微不足道。我很久没有想过在使用NIO之前我们必须如何做这件事;在这之前没有真正的非阻塞读取,或者aselect()
。使用setSoTimeout()您走在了正确的轨道上
因为这是你唯一的途径。你应该能够在从套接字读取之前调用它,如果它超时,它会抛出一个中断异常
。但是相当混乱-如果你可以使用NIO,我可以给你一个例子说明它是如何工作的。你是在处理到服务器的多个套接字连接,还是只处理一个?O实际上,它抛出了一个java.net.SocketTimeoutException
,只是尝试了一下。