BufferedReader.readLine()的坑

今天在跟朋友讨论实现服务器和客户端一对一通信的时候,客户端给服务器发送第一条数据无异常,接着服务器给客户端回数据的时候,怎么着客户端也收不到服务器的数据,调试后发现原来是卡在服务端的readLine() 方法上了

直接上代码

客户端:

import java.io.*;
import java.net.Socket;
import java.util.Scanner;

public class Client {

    public static void main(String[] args) {
        Socket socket = null;
        PrintWriter printWriter = null;
        BufferedReader bufferedReader = null;
        try {

            //创建套接字
            socket = new Socket("localhost", 9999);
            while (true) {
                // 给服务器发送数据
                System.out.println("给服务器发送数据");
                Scanner scanner = new Scanner(System.in);
                String sendInfo = scanner.nextLine();
                printWriter = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()));
                printWriter.println(sendInfo);
                printWriter.flush();
                if ("end".equals(sendInfo)) return;

                // 接收服务端数据
                bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
                String serverInfo = bufferedReader.readLine();
                if ("end".equals(serverInfo)) return;
                System.out.println("服务器:" + serverInfo);
            }
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            // 关闭流
            try {
                if (null != socket) {
                    socket.close();
                }
                if (null != printWriter) {
                    printWriter.close();
                }
                if (null != bufferedReader) {
                    bufferedReader.close();
                }

            } catch (IOException e) {
                e.printStackTrace();
            }
        }

    }

}

服务端

package org.net;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Scanner;

public class Server {
    public static void main(String[] args) {
        ServerSocket serverSocket = null;
        Socket socket = null;
        BufferedReader bufferedReader = null;
        try {
            System.out.println("等待客户端的连接");
            // 创建Server套接字
            serverSocket = new ServerSocket(9999);
            //接收 客户端
            socket = serverSocket.accept();
            bufferedReader = new BufferedReader(new InputStreamReader(socket.getInputStream()));
            while (true) {
                // 读取客户端数据
                String clientInfo = null;
                while (null != (clientInfo = bufferedReader.readLine())) {
                    System.out.println("客户端:" + clientInfo);
                }
                if ("end".equals(clientInfo)) return;

                //给客户端发送数据
                System.out.println("给客户端发送数据");
                Scanner scanner = new Scanner(System.in);
                String sendInfo = scanner.nextLine();
                PrintWriter printWriter = new PrintWriter(new OutputStreamWriter(socket.getOutputStream()));
                printWriter.println(sendInfo);
                printWriter.flush();
                if ("end".equals(sendInfo)) return;
            }

        } catch (IOException e) {
            e.printStackTrace();
        } finally {

            // 关闭连接
            try {
                if (serverSocket != null) {
                    serverSocket.close();
                }

                if (socket != null) {
                    socket.close();
                }

                if (bufferedReader != null) {
                    bufferedReader.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

    }
}

经过和朋友的讨论以及查资料发现,原来是对nextLine()的理解错误

我们错误的理解为:nextLine()在没有接收到客户端和读完客户端的数据时返回null

while (null != (clientInfo = bufferedReader.readLine())) {
       System.out.println("客户端:" + clientInfo);
}

其实不然

正解:

  1. readLine() 方法在没有接收到客户端的请求是一直处于阻塞状态,并没有返回null。
  2. 如果readLine()方法在没有读取到回车符时,也是一直等待读取,直到读取到回车符。
  3. 如果读到了回车符,他并没返回null,而是又重新进入阻塞状态等待数据的读取。
  4. 只有在连接关闭和发生资源错误的时候nextLine()才会返回null。
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
实验1 分析成绩单 1. 实验目的:掌握字符输入、输出流用法。 2. 实验代码: Fenxi: import java.util.*; public class Fenxi{ public static double getTotalScore(String s){ Scanner scanner=new Scanner(s); scanner.useDelimiter("[^0123456789.]+"); double totalScore=0; while(scanner.hasNext()){ try{ double score=scanner.nextDouble(); totalScore=totalScore+score; } catch(InputMismatchException exp){ String t=scanner.next(); } } return totalScore; } } AnalysisResult: import java.io.*; import java.util.*; public class AnalysisResult{ public static void main(String args[]){ File fRead=new File("score.txt"); File fWrite=new File("scoreAnalysis.txt"); try{ Writer out= new FileWriter(fWrite,true);//以尾加方式创建指向文件fWrite的out流 BufferedWriter bufferWrite=new BufferedWriter(out); //创建指向out的bufferWrite流 Reader in=new FileReader(fRead); //创建指向文件fRead的in流 BufferedReader bufferRead=new BufferedReader(in); //创建指向in的bufferRead流 String str=null; while ((str=bufferRead.readLine())!=null){ double totalScore =Fenxi.getTotalScore(str); str=str+"总分:"+totalScore; System.out.println(str); bufferWrite.write(str); bufferWrite.newLine(); } bufferRead.close(); bufferWrite.close(); } catch(IOException e){ System.out.println(e.toString()); } } } 3. 结果截图: 4. 实验分析: 1. 改进程序,使得能统计出每个学生的平均成绩。 答: 2. 现在有如下格式的货物明细(文本格式)goods.txt 品名:电视,length:102 cm,width:89 cm,height:56 cm. 品名:轿车,length:4502 cm,width:178 cm,height:156 cm. 品名:桌子,length:125 cm,width:78 cm,height:68 cm. 答: CalculateVolume.java import java.io.*; import java.util.*; public class CalculateVolume{ public static void main(String args[]){ File fRead=new File("goods.txt"); File fWrite=new File("goodsVolume.txt"); try{Writer out=new FileWriter(fWrite,true); BufferedWriter bufferWrite=new BufferedWriter(out); Reader in=new FileReader(fRead); BufferedReader bufferRead=new BufferedReader(in); String str=null; while((str=bufferRead.readLine())!=null){ double s=Jisuan.getVolume(str); str=str+"体积:"+s+"cm^3"; System.out.println(str); bufferWrite.write(str); bufferWrite.newLine(); } bufferRead.close()
`BufferedReader.readline()`是Java中可用于从输入流中读取一行文本的方法。它被称为"阻塞"是因为当调用这个方法时,如果没有可用的数据可以读取,程序将会被阻塞,即暂停执行直到有数据可用或者遇到异常。 当没有可用的数据时,`BufferedReader.readline()`方法将等待输入流中有足够的数据可供读取。这通常发生在以下情况下: 1. 输入流没有数据可供读取。例如,当程序正在读取一个空文件或者网络连接没有收到数据时,`BufferedReader.readline()`方法将一直阻塞。 2. 输入流的数据速度比程序读取的速度快。当程序没有及时读取输入流中的数据,输入流可能会积累一定量的数据。当程序准备好读取时,`BufferedReader.readline()`方法将会读取所有可用的数据行。 3. 输入流的数据被缓冲。在某些情况下,输入流可能会被缓冲以提高读取性能。当调用`BufferedReader.readline()`时,如果还没有足够的数据行在缓冲区中,此方法将阻塞直到有足够的数据行可供读取。 需要注意的是,阻塞是一种同步的行为,这意味着当程序在执行`BufferedReader.readline()`时会暂停执行,直到特定条件满足或者发生异常。因此,在使用`BufferedReader.readline()`方法时,我们需要注意处理可能的阻塞情况,例如使用多线程处理输入流或者设置适当的超时时间来避免长时间的阻塞。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

库里的球衣

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值