[JAVA]Socket中BufferedReader.readLine()的阻塞特性导致的数据无法多次发送的问题

我在写一个Socket服务器时,使用了BufferedReader.readLine()的方法来读取消息,客户端用BufferedWriter.write()来写数据,但是却出现了以下情况:


我的服务器接收代码为:

try{
	in = new BufferedReader(new InputStreamReader(socket.getInputStream()));//Java流读写
	out = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
	while(true){
		if(in.ready()){
			String msg=in.readLine();
			System.out.println(socket.getPort()+"-发来消息:\n"+msg);
		}
	}
}catch(Exception e){
	e.printStackTrace();
	return ;
}

客户端发送代码为:

try{
     BufferedWriter out = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
     out.write("FFi");

}catch (Exception e){
     e.printStackTrace();
}
可以注意到我发送的是"FFi"这个单元,但是在服务器接收到的却是"FFiFFiFFi",而且是在Scoket断开时一次性刷出来的。


我找了很久的原因,后来才知道readline()函数其实自带了一个阻塞....也就是说in.readline()实际上收到第一次"FFi"之后,依旧堵塞着,这条信息一直存在缓存里,下面的System.out.println()语句压根没执行到,直到Socket断开,强制清缓存的时候,readline()不再阻塞。


那应该怎么办呢?

其实readline()读取的结束标识符是"\n",所以客户端代码改成如下形式就好了:

try{
     BufferedWriter out = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
     out.write("FFi\n");//修改这里

}catch (Exception e){
     e.printStackTrace();
}
效果如图所示:



BufferedReaderreadLine()函数是一个阻塞函数,当没有数据可读取时,它会一直阻塞在那里,而不是返回null。这可能导致程序卡死的情况发生。为了避免这种情况,我们可以使用以下方法来解决: 1. 设置超时时间:可以使用setSoTimeout()方法设置读取超时时间,当超过指定时间后,readLine()函数会抛出SocketTimeoutException异常,从而避免程序卡死。 ```java BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream())); socket.setSoTimeout(5000); // 设置超时时间为5秒 String line = reader.readLine(); // 读取数据 ``` 2. 使用非阻塞模式:可以将Socket设置为非阻塞模式,这样readLine()函数在没有数据可读取时会立即返回null,而不会阻塞。 ```java SocketChannel channel = SocketChannel.open(); channel.configureBlocking(false); // 设置为非阻塞模式 channel.connect(new InetSocketAddress("localhost", 8080)); BufferedReader reader = new BufferedReader(new InputStreamReader(channel.socket().getInputStream())); String line = reader.readLine(); // 读取数据,如果没有数据可读取,会立即返回null ``` 3. 使用available()方法判断是否有数据可读取:可以使用available()方法判断输入流是否有数据可读取,如果没有数据可读取,可以暂停一段时间后再次尝试读取。 ```java BufferedReader reader = new BufferedReader(new InputStreamReader(socket.getInputStream())); while (true) { if (reader.ready()) { String line = reader.readLine(); // 读取数据 break; } else { Thread.sleep(1000); // 暂停1秒后再次尝试读取 } } ```
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值