我想创建能够与多个客户端连接的服务器,但我总是得到'Socket Closed'异常或从输入流中读取NULL值,之前我认为这是由于以错误的方式关闭套接字连接引起的,所以我发布了这个topic,但现在看来这不是问题。
服务器方法
//here I use thread pool to solve concurrency issue
public static void main(String[] args) {
try{
ExecutorService service = Executors.newFixedThreadPool(4);
ServerSocket serverSocket = new ServerSocket();
while (true) {
//keep listening
Socket socket = serverSocket.accept();
service.execute(new HandlerThread(socket));
}
}catch(Exception ex){
ex.printStackTrace();
}
}
HandlerThread类
public class HandlerThread extends Thread{
private Socket socket;
private BufferedReader in;
public HandlerThread(Socket socket) throws IOException {
this.socket = socket;
this.start();
}
public void run(){
try {
//Caches is a singleton class, which used to store the received data from client
Caches caches = Caches.getInstance();
//32 line
in = new BufferedReader(new InputStreamReader(socket.getInputStream(), "UTF-8"));
String line = in.readLine();
//caches.list is an ArrayList
synchronized (caches.globalCaches) {
if (null != line) {
caches.globalCaches.add(line);
}
}
System.out.println(line);
in.close();
socket.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
客户端模拟器
public class Main {
public static void main(String[] args) throws IOException, InterruptedException {
for(int i = 0 ; i < 40; i++){
Thread.sleep(2000);
test test1 = new test();
test1.start();
}
}
public static class test extends Thread{
public void run(){
try {
Socket socket = new Socket("localhost", 5050);
PrintStream out = new PrintStream(socket.getOutputStream());
out.println("Hello Server!");
out.close();
socket.close();
} catch (UnknownHostException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
例外与问题
在客户端模拟器之前启动服务器,客户端模拟器没有任何异常,但是在服务器端,并非所有数据都被接收,从控制台打印信息,我看到'null','Hello Server','java。 net.SocketException:Socket已关闭......',如下所示
Hello Server
null
null
Hello Server
java.net.SocketException: Socket is closed
Hello Server
at java.net.Socket.getInputStream(Socket.java:876)
at com.icubeidea.core.threads.HandlerThread.run(HandlerThread.java:32)
Hello Server
我研究了很长一段时间,仍然找不到究竟是什么导致了这些异常和数据丢失的问题,还有什么潜在的风险可能导致服务器运行一段时间后内存泄漏问题?
答案 0 :(得分:3)
问题似乎出现在HandlerThread
:
你打电话给this.start();
你真的不应该这样做。特别是在同时使用ExecutorService
时!
(在这种特定情况下,您正在创建一个新线程并start()
它,同时告诉您的执行者运行相同的代码。最后,将运行相同的代码<单个套接字上的em>两次,这可能会导致各种问题,其中一个线程将关闭套接字,而另一个线程仍在尝试从中读取。这实际上是异常的地方来自。)
解决方案:
不扩展线程,但让HandlerThread
仅实现Runnable
接口。
一旦将该类的实例移交给执行程序,它将负责运行您的run()
方法;无需自己启动任何线程。