繁体   English   中英

java.net.Socket> InputStream> BufferedReader.read(char [])阻塞线程

[英]java.net.Socket > InputStream > BufferedReader.read(char[]) blocks thread

我正在尝试使用BufferedReader.read(char [])方法,而不是更简单但用途更广泛的BufferedReader.readLine()方法,以从服务器接收答案。 BufferedReader与BufferedOutputStream并行使用,在下面的代码中,read(char [])方法将阻止所有内容,最后一个控制台输出是“ 新缓冲区,等待读取”。

客户:

public class MessageSender extends Thread {
private String message;
MessageSender(String message) {
    this.message = message;
}

public void run() {
    try {
        Socket sk = new Socket("192.168.1.4", 3000);
        BufferedOutputStream bo = new BufferedOutputStream(sk.getOutputStream());
        bo.write(message.getBytes());
        bo.flush();

        char[] c = new char[100];

        BufferedReader br = new BufferedReader(new InputStreamReader(sk.getInputStream()));
        StringBuffer sb = new StringBuffer();
        System.out.println("new buffer, waiting to read.");
        int ix = 0;

        while (ix != -1) {
            ix = br.read(c);
            sb.append(new String(c));
        }

        String message = sb.toString();

        System.out.println("reply: " + message);

        sk.close();
    } catch (IOException e) {
        e.printStackTrace();
    } catch (Exception e) {
        e.printStackTrace();
    }
}

}

服务器:

public class MessageReceiver extends Thread {
    public void run() {
        try {
            ServerSocket ss = new ServerSocket(3000);
            System.out.println("server socket open");
            while (true) {
                Socket sk = ss.accept();
                System.out.println("new connection");
                BufferedReader br = new BufferedReader(new InputStreamReader(sk.getInputStream()));
                String line = br.readLine();
                System.out.println("received line: " + line);
                BufferedOutputStream bo = new BufferedOutputStream(sk.getOutputStream());
                bo.write("ack".getBytes()); bo.flush(); //bo.close();
                sk.close();
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

主要:

public class Main {

    public static void main(String[] args) {
        MessageReceiver mr = new MessageReceiver();
        mr.start();
        while (true) {
            String msg = new Scanner(System.in).nextLine();
            MessageSender ms = new MessageSender(msg+"");
            ms.start();
        }
    }
}

只要不调用BufferedReader.read,一切正常。 但是由于代码是现在,输出似乎没有发送到服务器。

更新

@JB Nizet回答,问题出在服务器脚本中,该脚本使用readLine()并等待EOL字符或连接结束。 因此,在客户端发送的消息中添加“ \\ n”可以解决死锁:

bo.write((message+"\n").getBytes());

当服务器接受来自客户端的连接时,它要做的第一件事是:

String line = br.readLine();

因此,它将阻塞直到客户端发送完整的文本行。 如果服务器读取了EOL字符,或者客户端关闭了流,则服务器仅知道该行已完成。

客户端启动时,它要做的第一件事是

bo.write(message.getBytes());

message是一行文本,没有任何终止。 然后客户做

ix = br.read(c);

因此它等待服务器的响应,服务器本身也在等待客户端的EOL。

您已经实现了网络死锁。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM