Java Socket异常:Socket Closed和null值

时间:2014-07-30 09:08:49

标签: java multithreading sockets

我想创建能够与多个客户端连接的服务器,但我总是得到'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

我研究了很长一段时间,仍然找不到究竟是什么导致了这些异常和数据丢失的问题,还有什么潜在的风险可能导致服务器运行一段时间后内存泄漏问题?

1 个答案:

答案 0 :(得分:3)

问题似乎出现在HandlerThread

的构造函数中

你打电话给this.start();你真的不应该这样做。特别是在同时使用ExecutorService时!

(在这种特定情况下,您正在创建一个新线程并start()它,同时告诉您的执行者运行相同的代码。最后,将运行相同的代码<单个套接字上的em>两次,这可能会导致各种问题,其中一个线程将关闭套接字,而另一个线程仍在尝试从中读取。这实际上是异常的地方来自。)

解决方案:

扩展线程,但让HandlerThread仅实现Runnable接口。

一旦将该类的实例移交给执行程序,它将负责运行您的run()方法;无需自己启动任何线程。