问题描述
accept()方法无限循环内的阻塞,无法退出while循环。
API
public Socket accept() throws IOException
侦听要连接到此套接字并接受它。 该方法将阻塞直到建立连接。
创建一个新的Socket
s
,如果有安全管理器,则使用s.getInetAddress().getHostAddress()
和s.getPort()
作为其参数来调用安全管理器的checkAccept
方法,以确保允许操作。 这可能会导致SecurityException。结果
新的Socket
异常
IOException
- 如果在等待连接时发生I / O错误。
SecurityException
- 如果安全管理器存在,并且其checkAccept
方法不允许操作。
SocketTimeoutException
- 如果先前使用setSoTimeout设置了超时并且达到超时。
IllegalBlockingModeException
- 如果此套接字具有关联的通道,则该通道处于非阻塞模式,并且没有准备接受的连接另请参见:
public void setSoTimeout(int timeout) throws SocketException
启用/禁用
SO_TIMEOUT
带有指定超时,以毫秒为单位。 使用此选项设置为非零超时,对此ServerSocket的accept()的调用将仅阻止此时间。 如果超时超时, 则会引发java.net.SocketTimeoutException ,尽管ServerSocket仍然有效。 必须先启用该选项才能进入阻止操作才能生效。 超时时间必须为> 0
。 超时为零被解释为无限超时。参数
timeout
- 指定的超时,以毫秒为单位异常
SocketException
- 如果底层协议有错误,如TCP错误。从以下版本开始:
JDK1.1
另请参见:
解决方案
使用Timeout(int timeout) 设置超时时间
设置ServerSocket是否开启监听标记
package netchat;
import java.io.*;
import java.net.*;
import javax.swing.JTextArea;
public class NetSocket {
private ServerSocket ssk;
private boolean isStop;
public NetSocket() {
isStop=false;
ssk=null;
}
/**
* @param port 监听端口
* @throws IOException 如果发生I / O错误
*/
public void startReceive(int port) throws IOException{
// TODO 自动生成的方法存根
ssk = new ServerSocket(port);
isStop=false;
ssk.setSoTimeout(5*10000);
new Thread(()-> {
while(!isStop) {
Socket sk=null;
try {
sk = ssk.accept();
} catch (IOException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
new Thread(new ReceiveThread(sk)).start();
}
try {
ssk.close();
} catch (IOException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
}).start();
}
/**
* Close ReceiveThread
*/
public void stopReceive() {
// TODO 自动生成的方法存根
isStop=true;
}
}
参考文章
https://bbs.csdn.net/topics/250016069
https://blog.csdn.net/budapest/article/details/6941903
https://blog.csdn.net/weixin_41715077/article/details/88791499