Java - чтение из буферизованного читателя (из сокета) приостанавливает поток - PullRequest
7 голосов
/ 13 июня 2011

У меня есть поток, который читает символы из буферизованного читателя (созданного из сокета следующим образом):

inputStream = new BufferedReader(new InputStreamReader(clientSock.getInputStream()));

Этот код работает только один раз. Например, если клиент подключается и отправляет это: «Это тест» и «Это еще один тест», вывод хоста:

 Reading from stream:
 Chars read from stream: 16
 This is a test

 Reading from stream:

Обратите внимание, что программа не получает "Это еще один тест", потому что она застряла при чтении потока. Есть ли способ справиться с этим без уменьшения размера буфера? Это код для потока:

public void run() {
        boolean dataRecieved = false;
        char[] inputChars = new char[1024];
        int charsRead = 0;

        while (!stopNow) {

            try {
                Thread.sleep(getDataDelay);

                //Read 1024 characters. Note: This will pause the thread when stream is empty.
                System.out.println("Reading from stream:");
                charsRead =  inputStream.read(inputChars); //<< THIS LINE IS PAUSING THE THREAD!> 


                if ((charsRead =  inputStream.read(inputChars)) != -1)
                {
                    System.out.println("Chars read from stream: " + charsRead);  
                    System.out.println(inputChars);
                    System.out.flush();
                }


            } catch (IOException e) {
                System.out.println("IOException");
                //TODO: CLIENT HAS DISCONNECTED...
            } catch (InterruptedException e) {
                System.out.println("Interrupted");
                // Sleep was interrupted.
            } 

        }

    }

Код для клиента / отправителя (не мой код):

public static void main(String[] args) throws IOException {
        // <<<<<<<<<<< CLIENT >>>>>>>>>>>>>>>

        Socket sock = new Socket("127.0.0.1", 3000);
        // reading from keyboard (keyRead object)
        BufferedReader keyRead = new BufferedReader(new InputStreamReader(System.in));
        // sending to client (pwrite object)
        OutputStream ostream = sock.getOutputStream(); 
        PrintWriter pwrite = new PrintWriter(ostream, true);

        // receiving from server ( receiveRead  object)
        InputStream istream = sock.getInputStream();
        BufferedReader receiveRead = new BufferedReader(new InputStreamReader(istream));

        System.out.println("Start the chitchat, type and press Enter key");

        String receiveMessage, sendMessage;               
        while(true)
        {
            sendMessage = keyRead.readLine();     // keyboard reading
            pwrite.println(sendMessage);       // sending to server
            System.out.flush();         // flush the data

            if((receiveMessage = receiveRead.readLine()) != null) //receive from server
            {
                System.out.println(receiveMessage); // displaying at DOS prompt
            }         
        }               
    }          

Ответы [ 5 ]

10 голосов
/ 13 июня 2011

java.io.InputStream.read() - это блокирующий вызов , что означает, что если данные недоступны, поток останавливается до тех пор, пока данные не станут доступны.

Для неблокирующего ввода-вывода используйте классыjava.nio пакет.

2 голосов
/ 13 июня 2011

Ваш «отправитель» ожидает получения данных от «получателя», и именно здесь код ожидает бесконечно. Предполагается, что получатель отправляет ответ, когда получает сообщение?

0 голосов
/ 05 ноября 2015
Socket socket;

// Assuming socket is connected and not null

if(socket != null){
    if(socket.getInputStream().available() > 0){
        byte[] buffer;
        buffer = new byte[socket.getInputStream().available];
        socket.getInputStream().read(buffer);

        // Your code here to deal with buffer.

    }
}

Если вы хотите записать в сокет,

OutputStream mmOutStream;
mmOutStream = socket.getOutputStream();

public void write(byte[] buffer) {
    try {
        mmOutStream.write(buffer);
    } catch (IOException e) {
        Log.e(TAG, "Exception during write ", e);
    }
}
0 голосов
/ 29 июня 2015

Реализуйте протокол, в котором вы отправляете длину ваших данных в заголовки, чтобы сервер / клиент знал, сколько данных ожидать.

0 голосов
/ 13 июня 2011

Вы должны создать ServerSocket Это прослушивает клиента в каждом цикле.

ServerSocket socket = new ServerSocket(3000);

Вот мой метод run (), который будет ждать клиента Socket каждый раз

public void run(){
        boolean dataRecieved = false;
        char[] inputChars = new char[1024];
        int charsRead = 0;

        while (!stopNow) {
            try {
                System.out.println("Listen To Clients:");

                // The ServerSocket has to listen the client each time.
                InputStreamReader isr = new InputStreamReader( socket.accept().getInputStream() );
                inputStream = new BufferedReader( isr );

                //Read 1024 characters. Note: This will pause the thread when stream is empty.
                System.out.println("Reading from stream:");

                if ((charsRead =  inputStream.read(inputChars)) != -1)
                {
                    System.out.println("Chars read from stream: " + charsRead);  
                    System.out.println(inputChars);
                    System.out.flush();
                }
            } 
            catch (IOException e) 
            {
                e.printStackTrace();
            }
        }
    }

У вас есть еще одна незначительная ошибка, которая останавливает код и удаляет строку

charsRead =  inputStream.read(inputChars); //<< THIS LINE IS PAUSING THE THREAD!>

Поскольку эта строка перемещается в операторе if.

...