【Java】Java socket通信使用read,readline函数的阻塞问题
Socket通信是Java网络编程中比较基础的部分,其原理其实就是源ip,源端口和目的ip,目的端口组成的套接字通信。其底层还设及到了TCP协议的通信。
Java中的Socket通信可以通过客户端的Socket与服务端的ServerSocket通信,同时利用IO流传递数据,也就是说Socket通信是面向流的使用的是BIO,并不同于后来的NIO通信面向缓冲。Socket通信中使用的IO流的read,readline等函数都是阻塞的,这就导致了在通信过程中,双方不能确定什么时侯是流的结束,针对这种可以通过约定结束符的方式进行结束,也可以约定一次传输的字节流的长度。下面通过代码进行说明
客户端
建立客户端线程,在run方法中不断对服务端进行发送消息,模拟多个客户端的通信,通过写入换行符,表明这次通信的结束。
1 class Client implements Runnable {
2
3 private byte[] targetIp;
4 private int port;
5
6 Client(byte[] ip, int port) {
7 this.targetIp = ip;
8 this.port = port;
9 }
10
11 @Override
12 public void run() {
13 try {
14 InetAddress inetAddress = InetAddress.getByAddress(targetIp);
15 Socket socket = new Socket(inetAddress, port);
16 System.out.println("client");
17 BufferedReader socketInput = new BufferedReader(new InputStreamReader(socket.getInputStream()));
18 BufferedWriter socketOutput = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
19
20 int i = 0;
21 String NAME = "Client";
22 while (true) {
23 socketOutput.write("This msg from " + NAME + " msg id is " + i);
24 socketOutput.write("\n");//约定结束符表示流的结束
25 i++;
26 socketOutput.flush();
27 System.out.println("here");
28 String str = null;
29 if (!(str = socketInput.readLine()).equals("\n")) {
30 System.out.println(str);
31 }
32
33 }
34
35 /*socket.shutdownInput();
36 socket.shutdownOutput();*/
37 } catch (IOException e) {
38 e.printStackTrace();
39 }
40
41 }
42 }
服务端
服务端通过accept接受客户端的连接,这个操作是阻塞的,直到有客户端的连接才能进行下一步。
1 class Server implements Runnable {
2
3 private int port;
4
5 Server(int port) {
6 this.port = port;
7 }
8 @Override
9 public void run() {
10 try {
11 ServerSocket serverSocket = new ServerSocket(port);
12
13 InetAddress inetAddress = serverSocket.getInetAddress();
14 System.out.println("server" + inetAddress.getHostAddress());
15 Socket socket = serverSocket.accept();
16 BufferedReader socketInput = new BufferedReader(new InputStreamReader(socket.getInputStream()));
17 BufferedWriter socketOutput = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));
18
19 int i = 0;
20 while (true) {
21 String str = null;
22 if (!(str = socketInput.readLine()).equals("\n")) System.out.println(str);
23 System.out.println("server");
24
25 String NAME = "Server";
26 socketOutput.write("This msg from " + NAME + " msg num is " + i + " reply to " + str);
27 socketOutput.write("\n");
28 i++;
29 socketOutput.flush();
30 }
31
32 // socket.shutdownInput();
33 // socket.shutdownOutput();
34 } catch (IOException e) {
35 e.printStackTrace();
36 }
37
38 }
39 }
测试
1 public class SocketTest {
2 public static void main(String[] args) {
3 byte[] ip = {127, 0, 0, 1};
4 int port = 27149;
5 Thread server = new Thread(new Server(port), "server");
6 server.start();
7 Thread client = new Thread(new Client(new byte[]{0,0,0,0}, 27149));
8 client.start();
9 }
10 }
结果
服务端针对客户端的每条信息都能够进行读取并返回消息给客户端,但是如果注释掉写入换行符,并判断读取是否是换行符的代码,就无法读取到流的结束。
【Java】Java socket通信使用read,readline函数的阻塞问题的更多相关文章
- Flex通信-与Java实现Socket通信实例
Flex通信-与Java实现Socket通信实例 转自:http://blessht.iteye.com/blog/1136888 博客分类: Flex 环境准备 [服务器端] JDK1.6,“ja ...
- java的socket通信
本文讲解如何用java实现网络通信,是一个非常简单的例子,我比较喜欢能够立马看到结果,所以先上代码再讲解具体细节. 服务端: import java.io.BufferedReader; import ...
- Java的Socket通信----通过 Socket 实现 TCP 编程之多线程demo(2)
JAVA Socket简介 所谓socket 通常也称作”套接字“,用于描述IP地址和端口,是一个通信链的句柄.应用程序通常通过”套接字”向网络发出请求或者应答网络请求. import java.io ...
- Java的Socket通信简单实例
服务端 package testlxd; import java.io.BufferedReader; import java.io.IOException; import java.io.Input ...
- C++服务器与java进行socket通信案例
分类: [java]2012-10-08 12:03 14539人阅读 评论(46) 收藏 举报 注:本代码版权所有!!!转载时请声明源地址:http://blog.csdn.net/nuptboyz ...
- Java:Socket通信
Socket通常也称作"套接字".用于描写叙述IP地址和port,是一个通信链的句柄.应用程序通常通过"套接字"向网络发出请求或者应答网络请求. ServerS ...
- java的Socket通信例子及关于java.net.SocketException: Socket is closed错误
今天写socket数据相互通信的时候,碰到一个及其蛋疼的错误.单向传输数据的时候server与client是没有问题的,但是两个都有输入输出操作的时候就出现了这个问题 java.net.SocketE ...
- java NIO socket 通信实例
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/zhuyijian135757/article/details/37672151 java Nio 通 ...
- Java实现Socket通信
一对一通信: 服务器端: public class ServerDemo { public static void main(String[] args) { ServerSocket serverS ...
随机推荐
- SQL Server 列存储索引 第三篇:维护
列存储索引分为两种类型:聚集的列存储索引和非聚集的列存储索引,在一个表上只能创建一个聚集索引,要么是聚集的列存储索引,要么是聚集的行存储索引,然而一个表上可以创建多个非聚集索引. 一,创建列存储索引 ...
- MongoDB下载安装与配置
一.下载与安装 1.进入mongodb官网,try->download->community找到下载.或者直接在地址栏输入:https://www.mongodb.com/try/down ...
- CF1137A/1138C Skyscrapers
排序+数据结构 将每一行(每一列)都排个序,并将原位置的在这一行(列)中的排行记录在一个数组里 注意,要将楼高度相同的元素看作一个元素 如 1 1 4 5 6 8 8,则排行是 1 1 2 ...
- SVG--D3--血缘关系树
最近的工作与可视化有关,有展示血缘关系树的需求 ,类似于这样: 碰巧搜到 D3(用于可视化的js库,作者吕之华),瞬间无法自拔,它的树状图功能基于SVG.js ,暴露的可操作入口也简洁恰当,能帮助你快 ...
- tensorflow-gpu2.1.0报错 so returning NUMA node zero解决办法
>>> print('Default GPU Device: {}'.format(tf.test.gpu_device_name()))2020-06-06 10:14:08.92 ...
- leetcode1Minimum Depth of Binary Tree
题目描述 求给定二叉树的最小深度.最小深度是指树的根结点到最近叶子结点的最短路径上结点的数量. Given a binary tree, find its minimum depth.The mini ...
- IL角度理解for 与foreach的区别——迭代器模式
IL角度理解for 与foreach的区别--迭代器模式 目录 IL角度理解for 与foreach的区别--迭代器模式 1 最常用的设计模式 1.1 背景 1.2 摘要 2 遍历元素 3 删除元素 ...
- 性能工具-io工具
I/O:某网上问题通过top iotop pidstat vmstat 工具定位出io高原因,内存不够.
- spring boot实现超轻量级网关(反向代理、转发)
在我们的rest服务中,需要暴露一个中间件的接口给用户,但是需要经过rest服务的认证,这是典型的网关使用场景.可以引入网关组件来搞定,但是引入zuul等中间件会增加系统复杂性,这里实现一个超轻量级的 ...
- CSS属性(边框)
1.边框 <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="U ...