我有一个在 Windows XP 和 Windows 7 上运行的 java 应用程序。这个应用程序与另一台计算机有一个开放的 tcp 连接。
现在重点来了:在 XP 机器上,通过返回 -1 或 SocketException,拔出网络电缆对我的 inputstream.read 方法产生了几乎直接的影响。这是我检测网络连接丢失的一件事。在 Win7 机器上拔下网络电缆时,Windows 本身在任务栏中的连接图标中告诉我这一点,但我的 Java 应用程序没有得到任何提示。就好像什么都没发生过一样。
TCP 在 Windows 7 上的行为是否不同,或者 Windows7 是否会从其应用程序中屏蔽此信息?
package tcpconnectionlossdetectiontest;
import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketException;
import java.net.SocketTimeoutException;
public class TcpListener extends Thread {
private int _portNo = 23456;
private boolean _done = false;
public static void main(String[] args) {
int port = 23456;
if (args.length > 0) {
int position = -1;
do {
String arg = args[++position].trim();
if (arg.equalsIgnoreCase("-p") || arg.equalsIgnoreCase("--PortNo")) {
String argParameter = args[++position].trim();
// e.g. store additional argument
try {
port = Integer.parseInt(argParameter);
} catch (NumberFormatException nfex) {
port = 23456;
}
System.out.println("Argument " + position + " (" + arg + ", " + argParameter + "): port number set to " + port);
} else {
System.out.println("Argument " + position + " (" + arg + ") unknown.");
}
}
while (position + 1 < args.length);
// Parsing command line arguments ready.
}
TcpListener listener = new TcpListener(port);
listener.start();
}
public TcpListener(int portNo) {
this._portNo = portNo;
}
public void run() {
Socket s = null;
InputStream is = null;
byte[] buf = new byte[1000];
int readResult = 0;
int maxOpen = 3;
int openCounter = 0;
try {
ServerSocket sSocket = new ServerSocket(this._portNo);
while (openCounter < maxOpen) {
if (s == null) {
try {
System.out.println("waiting for connection on port " + this._portNo);
sSocket.setSoTimeout(60000);
s = sSocket.accept();
if (s != null) {
System.out.println("got connection on port " + this._portNo);
openCounter++;
is = s.getInputStream();
}
} catch (SocketTimeoutException stex) {
System.out.println("no connection yet...");
}
}
if (s != null && is != null) {
readResult = is.read(buf, 0, 1000);
}
if (readResult == -1) {
readResult = 0;
System.out.println("connection broken...");
is=null;
if (s != null) {
s.close();
s=null;
}
} else if (readResult > 0) {
System.out.println("Data read: " + new String (buf,0,readResult));
}
}
} catch (IOException ioex) {
System.out.println("IO exception caught:");
ioex.printStackTrace();
}
this._done = true;
}
}
大约 5 秒后,windows xp 上的结果是断开连接后:
waiting for connection on port 23456
no connection yet...
waiting for connection on port 23456
got connection on port 23456
Data read: Here is a connection from win7 to xp...
Data read:
IO exception caught:
java.net.SocketException: Connection reset
at java.net.SocketInputStream.read(SocketInputStream.java:168)
at tcpconnectionlossdetectiontest.TcpListener.run(TcpListener.java:73)
Process exited with exit code 0.
断开连接后在 Windows 7 上的结果是:...什么都没有。应用程序在没有断开检测的情况下保持打开状态。
@edit 2013-02-01 09:54:发生在 jre6 和 jre7 中。
@edit 2013-02-01 10:59:发生在 32 位和 64 位 java 中。发生在 Win7 笔记本电脑和普通 PC 上。
@edit 2013-02-01 12:40:添加了代码示例和结果。
@edit 2031-02-06:由于我在互联网上一无所获,也没有人对此有解决方案,因此我在我们的应用程序中添加了一个主动乒乓机制。现在我最多可以在 20 秒后检测到。连接断开。
@edit 2013-02-08:另一种了解状态的方法是“netsh interface ip show interfaces”,但是在知道状态之前必须在 java 中解析命令行结果。也许有更好的方法来获取这些信息。