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函数的阻塞问题的更多相关文章

  1. Flex通信-与Java实现Socket通信实例

    Flex通信-与Java实现Socket通信实例  转自:http://blessht.iteye.com/blog/1136888 博客分类: Flex 环境准备 [服务器端] JDK1.6,“ja ...

  2. java的socket通信

    本文讲解如何用java实现网络通信,是一个非常简单的例子,我比较喜欢能够立马看到结果,所以先上代码再讲解具体细节. 服务端: import java.io.BufferedReader; import ...

  3. Java的Socket通信----通过 Socket 实现 TCP 编程之多线程demo(2)

    JAVA Socket简介 所谓socket 通常也称作”套接字“,用于描述IP地址和端口,是一个通信链的句柄.应用程序通常通过”套接字”向网络发出请求或者应答网络请求. import java.io ...

  4. Java的Socket通信简单实例

    服务端 package testlxd; import java.io.BufferedReader; import java.io.IOException; import java.io.Input ...

  5. C++服务器与java进行socket通信案例

    分类: [java]2012-10-08 12:03 14539人阅读 评论(46) 收藏 举报 注:本代码版权所有!!!转载时请声明源地址:http://blog.csdn.net/nuptboyz ...

  6. Java:Socket通信

    Socket通常也称作"套接字".用于描写叙述IP地址和port,是一个通信链的句柄.应用程序通常通过"套接字"向网络发出请求或者应答网络请求. ServerS ...

  7. java的Socket通信例子及关于java.net.SocketException: Socket is closed错误

    今天写socket数据相互通信的时候,碰到一个及其蛋疼的错误.单向传输数据的时候server与client是没有问题的,但是两个都有输入输出操作的时候就出现了这个问题 java.net.SocketE ...

  8. java NIO socket 通信实例

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/zhuyijian135757/article/details/37672151 java Nio 通 ...

  9. Java实现Socket通信

    一对一通信: 服务器端: public class ServerDemo { public static void main(String[] args) { ServerSocket serverS ...

随机推荐

  1. LOJ 6068「2017 山东一轮集训 Day4」棋盘

    题意 一个 \(n\times n\) 的棋盘上面有若干障碍物. 定义两个棋子可以互相攻击当且仅当这两个棋子的横坐标或纵坐标相等而且中间不能隔着障碍物.(可以隔棋子) 有 \(q\) 次询问,每次询问 ...

  2. k8s中pod的yaml文件全面解读

    apiVersion: v1 #必选,版本号,例如v1,版本号必须可以用 kubectl api-versions 查询到 . kind: Pod #必选,Pod metadata: #必选,元数据 ...

  3. python爬虫自定义header头部

    一.Handler处理器 和 自定义Opener 关注公众号"轻松学编程"了解更多. opener是 urllib.OpenerDirector 的实例,我们之前一直都在使用的ur ...

  4. IDEA 搭建 Spark 源码 (Ubuntu)

    版本:Spark 2.4.3/JDK 1.8/Scala 2.11.0 1.选择Spark版本.压缩包下载. 2.IDEA中左下角Terminal下输入: mvn -DskipTests clean ...

  5. GDB调试基础使用方法

    尽管目前使用的VS code可以使用插件一键构建和运行程序,但GDB作为调试利器,还是值得花时间去学习的. 概述 GDB(GNU Debugger) 是一个由GNU开源组织发布的.UNIX/LINUX ...

  6. c++11-17 模板核心知识(二)—— 类模板

    类模板声明.实现与使用 Class Instantiation 使用类模板的部分成员函数 Concept 友元 方式一 方式二 类模板的全特化 类模板的偏特化 多模板参数的偏特化 默认模板参数 Typ ...

  7. Newton插值的C++实现

    Newton(牛顿)插值法具有递推性,这决定其性能要好于Lagrange(拉格朗日)插值法.其重点在于差商(Divided Difference)表的求解. 步骤1. 求解差商表,这里采用非递归法(看 ...

  8. KOA2 笔记

    KOA2 基于ES7开发,完全使用Promise并配合async来实现异步的node框架 核心是对node的HTT模块P进行了封装,用多个async函数组成处理链,来不断地接收HTTP请求(ctx对象 ...

  9. java中在构造方法中修改线程名,修改失败原因(现已修改成功)

    package com.cx.Threads; public class HelloThread extends Thread{ @Override public void run() { for ( ...

  10. jquery DataTable插件使用自定义搜索

    $(function () { $("#pk_status").change(function () { valid = $(this).val(); if(valid){ tab ...