我正在尝试使用this教程和一些扩展来实现服务器(能够接受多个客户端)和客户端。 除了一个功能外,一切正常: 当服务器运行1分钟后,它会关闭,每个客户端都必须打印“服务器关闭”消息并关闭。
在我的模拟中KKMultiServer我用这种方式实现它,在主方法中:
long in1minute = 10*1000;
Timer timer = new Timer();
timer.schedule( new TimerTask(){
public void run() {
for(int i=0;i<startedThreads.size();i++){
try {
startedThreads.get(i).Close();
} catch (IOException e) {
e.printStackTrace();
}
}
listening=false;
}
}, in1minute );
当每个已启动的线程被添加到向量中时。
在我对KKMultiServerThread的模拟中,此代码应该将消息发送给客户端(我的模拟到KKProtocol正确进行)
public void Close() throws IOException{
String outputLine=p.processInput("shut");
out.println(outputLine);
out.close();
in.close();
socket.close();
}
服务器正常停止,但客户端不会打印“服务器关闭”消息。然后,如果我在服务器关闭时调用客户端,我会得到:
[java] Exception in thread "main" java.net.SocketException: Connection rese
tClient: d
[java] at java.net.SocketInputStream.read(SocketInputStream.java:189)
[java] at java.net.SocketInputStream.read(SocketInputStream.java:121)
[java] at sun.nio.cs.StreamDecoder.readBytes(StreamDecoder.java:283)
[java] at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:325)
[java] at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:177)
[java] at java.io.InputStreamReader.read(InputStreamReader.java:184)
[java] at java.io.BufferedReader.fill(BufferedReader.java:154)
[java] at java.io.BufferedReader.readLine(BufferedReader.java:317)
[java] at java.io.BufferedReader.readLine(BufferedReader.java:382)
[java] at ist.assignment2.Client.main(Client.java:31)
[java]
Client.java:31
的代码
while ((fromServer = in.readLine()) != null) {
...然后类似于KKClient
那么为什么客户不打印消息并退出?
答案 0 :(得分:2)
如果另一方未确认所有数据,则不应关闭TCP套接字。此机制适用于大多数应用程序协议。例如,对于HTTP,SMTP,IMAP,POP3,服务器在连接应该关闭时发出信号,客户端关闭套接字,然后服务器关闭套接字。
在您的情况下,您可以按如下方式扩展Close
方法:
public void Close() throws IOException {
String outputLine = p.processInput("shut");
out.println(outputLine);
// signal the client that the connection should be closed
socket.shutdownOutput();
// wait on ack by blocking until EOF is received
if (in.read() != -1) {
throw new RuntimeException("unexpected data sent by client");
}
// close socket
socket.close();
}