【发布时间】:2011-12-11 12:00:13
【问题描述】:
我目前正在调试两个通过 TCP 连接交换数据的 Java 应用程序。
其中一个应用程序 TCP 客户端通过调用 Socket#sendUrgentData(int) 定期向另一个应用程序 TCP 服务器发送紧急数据。第 18 次尝试发送紧急数据时,TCP 客户端抛出以下异常
java.io.IOException:BrokenPipe
at java.net.PlainSocketImpl.socketSendUrgentData(Native Method)
at java.net.PlainSocketImpl.sendUrgentData(PlainSocketImpl.java:541)
at java.net.Socket.sendUrgentData(Socket.java:927)
TCP 服务器抛出此异常
java.net.SocketException: Software caused connection abort: recv failed
at java.net.SocketInputStream.socketRead0(Native Method)
at java.net.SocketInputStream.read(Unknown Source)
at java.net.SocketInputStream.read(Unknown Source)
我相信异常是由尝试写入/读取关闭的连接/套接字引起的。我不明白为什么在调用 sendUrgentData() 17 次后连接或套接字会关闭。我能够重复它,它总是在 17 次之后发生。
如果我在 Windows 上运行客户端和服务器,就会出现问题。如果我在 Solaris 上运行客户端和服务器,则不会出现此问题。如果我在 Solaris 上运行客户端并在 Windows 上运行服务器,则会出现问题。如果我在 Windows 上运行客户端并在 Solaris 上运行服务器,则不会出现此问题。这让我觉得它可能与 Windows 相关?
使用 Wireshark 我在连接上看到以下流量
--> = from TCP client to TCP server
<-- = from TCP server to TCP client
--> [PSH, ACK, URG] (Seq=1, Ack=1)
<-- [ACK] (Seq=1, Ack=2)
--> [PSH, ACK, URG] (Seq=2, Ack=1)
<-- [ACK] (Seq=1, Ack=3)
...
--> [PSH, ACK, URG] (Seq=17, Ack=1)
<-- [RST, ACK] (Seq=1, Ack=18)
我写了一些简单的测试类来显示这个问题。
TCPServer.java IP_Address 端口
public class TCPServer
{
public static void main(String[] args) throws Exception
{
ServerSocket socket = new ServerSocket();
socket.bind(new InetSocketAddress(args[0], Integer.parseInt(args[1])));
System.out.println("BOUND/" + socket);
Socket connection = socket.accept();
System.out.println("CONNECTED/" + connection);
int b;
while ((b = connection.getInputStream().read()) != -1) {
System.out.println("READ byte: " + b);
}
System.out.println("CLOSING ..");
connection.close();
socket.close();
}
}
TCPClient.java IP_Address Port Interval_Between_Urgent_Data
public class TCPClient
{
public static void main(String[] args) throws Exception
{
final Socket socket = new Socket();
socket.connect(new InetSocketAddress(InetAddress.getByName(args[0]), Integer.parseInt(args[1])));
System.out.println("CONNECTED/"+socket);
Timer urgentDataTimer = new Timer(true);
urgentDataTimer.scheduleAtFixedRate(new TimerTask()
{
int n = 0;
public void run() {
try {
System.out.println("SENDING URGENT DATA ("+(++n)+") ..");
socket.sendUrgentData(1);
System.out.println("SENT URGENT DATA");
} catch (Exception e) {
e.printStackTrace();
}
}
}, 1000, Integer.parseInt(args[2]));
int b;
while ((b = socket.getInputStream().read()) != 1) {
System.out.println("READ byte: " + b);
}
System.out.println("CLOSING ..");
urgentDataTimer.cancel();
socket.close();
}
}
有人能解释一下这里发生了什么吗?
谢谢。
【问题讨论】:
标签: java windows sockets networking tcp