ObjectInputStream消耗太多内存

时间:2015-12-30 16:07:50

标签: java sockets

我有一个Socket,每隔几秒就会通过ObjectOutputStream向客户端发送一个对象列表。在服务器端,每writeObject(myList)后执行flush,然后执行reset。使用VisualVM检查内存使用情况,在服务器上没有内存泄漏,但在客户端上,似乎先前读取的列表保存在内存中。我尝试在客户端的ObjectInputStream上执行reset,但看起来像ObjectInputStream不支持此方法(它抛出java.io.IOException: mark/reset not supported)。

这是我的服务器套接字:

public class ConsultaBombas {

    public static void inicializarServidorSocket() {
        try {
            ServerSocket serverSocket = new ServerSocket(5963);
            Thread thread = new Thread(() -> {
                while (!serverSocket.isClosed()) {
                    try {
                        final Socket socket = serverSocket.accept();
                        new ThreadComunicacao(socket).start();
                    } catch (Exception e) {
                        e.printStackTrace();
                    }
                }
            });
            thread.setName("Consulta bombas (Inicializador)");
            thread.start();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    static class ThreadComunicacao extends Thread {
        private Socket socket;

        public ThreadComunicacao(Socket socket) {
            this.socket = socket;
            setName("Consulta bombas (Comunicação) com início: " + new SimpleDateFormat("dd/MM/yyyy HH:mm:ss").format(new Date()));
        }

        @Override
        public void run() {
            try {
                ObjectOutputStream out = new ObjectOutputStream(socket.getOutputStream());
                while (!socket.isClosed()) {
                    List<Bomba> bombas = new DaoBomba().findAll();
                    out.writeObject(bombas);
                    out.flush();
                    out.reset();
                    Thread.sleep(1000);
                }
            } catch (SocketException e) {
                if (e.getLocalizedMessage() != null && e.getLocalizedMessage().equalsIgnoreCase("Connection reset by peer: socket write error")) {
                    System.out.println("Cliente desconectou...");
                } else {
                    e.printStackTrace();
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

}

这是客户端(以start()方法启动):

public class ConsultaBombasClient {

    private Socket socket;
    private Thread threadConsulta;

    public ConsultaBombasClient(BombasListener bombasListener, String maquinaDestino) {
        threadConsulta = new Thread(() -> {
            try {
                Thread.currentThread().setName("Consulta Bombas");
                System.out.println("Endereço bagual: "+maquinaDestino);
                socket = new Socket(maquinaDestino, 5963);
                ObjectInputStream in = new ObjectInputStream(socket.getInputStream());
                Object leitura;
                while ((leitura = in.readObject()) != null) {
                    List<Bomba> bombas = (List<Bomba>) leitura;
                    bombasListener.run(bombas);
                }
            } catch (Exception e) {
                e.printStackTrace();
                throw new RuntimeException(e);
            }
        });
        threadConsulta.setDaemon(true);
    }

    public void start() {
        threadConsulta.start();
    }

    public interface BombasListener {
        void run(List<Bomba> bombas);
    }

}

我做错了什么?

2 个答案:

答案 0 :(得分:1)

垃圾收集不是即时的,你有任何真正的内存麻烦吗?您是否尝试过运行-Xmx值较低的客户端,是否收到了OutOfMemoryError?                      - user3707125

你是对的,在内存接近最大堆大小的一段时间后,它会从内存中清除对象。我没有看到这个,因为我的电脑里有很多内存但是使用Xmx50m我可以看到这个工作正如你所说。 - Mateus Viccari

答案 1 :(得分:0)

显然bombasListener.run(),无论它是什么,都不会发布提供的列表。

NB ObjectInputStream.readObject()在流结束时不返回null。因此,将此测试用作读取循环的终止条件是不正确的。