文章目录

  1. 前言
  2. Socket类
    2.1 Socket的通信过程
    2.2 控制Socket连接
    2.3 设置Socket的选项
  3. ServerSocket类
    3.1 构造ServerSocket
    3.2 ServerSocket的常用方法
    3.3 设置ServerSocket选项
  4. 基于TCP的BIO通信
    4.1 实现功能
    4.2 程序源代码
    4.3 程序效果截图

1. 前言

基于传输层协议TCP的网络通信是可靠的、有序的、差错控制的。TCP是面向连接的、可靠的流服务协议。TCP协议中,只有实现连接的双发才可以进行通信,因此广播和多播不是基于TCP的。下面首先介绍一些Socket和ServerSocket。
服务器、客户端建立通信的过程如下:

在这里插入图片描述

  1. 服务器网络接口的IP地址为10.1.1.1。服务器端应用程序在80端口运行服务,建立ServerSocket,并监听有无连接请求。
  2. 客户端必须知道服务器的IP地址或主机名、服务开放端口,客户端向服务器的服务Socket(10.1.1.1:80)发起请求。
  3. 服务器一旦接收该请求,就会在80端口上建立一个Socket,该Socket的本地地址为10.1.1.1:80,外部地址为客户端建立的Socket(client端IP地址,client端port)。客户端port通常由操作系统指定。
  4. 服务器每接收一个客户端的请求,就会在服务端口上建立一个Socket,这些Socket的本地地址相同,但外部地址各不相同。每个外部地址即各个客户端的本地Socket。这样服务器既可以监听服务请求,又可以和每个客户端进行独立的通信。
  5. 客户端通过本地Socket与服务器进行通信、读写信息。就像打电话一样,一个号码呼叫另一个号码,一旦被呼叫者接听,通信链路就建立起来了,双方就可以通话了。

2. Socket类

2.1 Socket的通信过程

在这里插入图片描述

2.2 控制Socket连接

获取Socket的输入流和输出流
服务端和客户端之间通信的Socket一旦连接,就会通过一对输入输出流进行通信。如同打电话时,一旦被叫放接听,双方通话的业务信道建立,就可以交流了。
Socket的getInputStream()方法和getOutputStream()方法分别实现获得输入流和输出流。

BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(socket.getOutputStream()));BufferedReader br = new BufferedReader(new IntputStreamReader(socket.getInputStream()));

获取Socket的地址端口信息
如果需要获得本地或者远程Socket的地址和端口信息,可以使用以下方法:

getInetAddress() //获得远程服务器的IP地址getPort() //获得远程服务器的端口getLocalAddress() //获得客户端本地的IP地址getLocalPort() //获得客户端本地的端口

关闭Socket
当客户端与服务器的通信结束时,应该关闭Socket,以释放Socket占用的包括

InputStream和OutputStream在内的各种资源。if(!s.isClosed()) s.close();//判断如果不是空的话则关闭

Socket的状态
Socket类提供了3中方法(isBound 、isClosed 和 isConnected)来判断Socket的状态。
isBound方法用来判断Socket的绑定状态,只要曾经绑定过,即使Socket已经关闭,仍然返回true。可以理解为本地是否曾经建立过到远程主机的Socket连接。
isClosed方法用来判断Socket是否已经关闭。
isConnected方法用来判断Socket的连接状态。和isBound方法一样,即使Socket已经关闭,仍然返回true,isConnected的状态不清楚。可以理解为到远程主机的Socket是否曾经连接过。

3. 基于TCP的BIO通信

3.1 实现功能

服务器创建服务并监听端口,与客户端建立连接,并向客户端转发消息,客户端输入quit命令即可退出。

3.2 程序源代码

Server类:

package com.company.bio;

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

public class Server {    public static void main(String[] args){        final  String QUIT = "quit";        final int DEFAULT_PORT = 8888;        ServerSocket serverSocket = null;

        //绑定监听端口        try{            serverSocket = new ServerSocket(DEFAULT_PORT);            System.out.println("启动服务器,监听端口"+DEFAULT_PORT);

            while (true){                // 等待客户端连接                Socket socket = serverSocket.accept();                System.out.println("客户端[" + socket.getPort() + "]已连接");                //创建IO流                BufferedReader bufferedReader = new BufferedReader(                        new InputStreamReader(socket.getInputStream())                );                BufferedWriter bufferedWriter = new BufferedWriter(                        new OutputStreamWriter(socket.getOutputStream())                );

                String msg = null;                while ((msg = bufferedReader.readLine()) != null) {                    // 读取客户端发送的消息,当对方关闭时返回null

                    System.out.println("客户端["+socket.getPort()+"]:"+ msg);                    //回复客户发送的消息                    bufferedWriter.write("服务器:" + msg + "\n");                    bufferedWriter.flush(); //保证缓冲区的数据发送出去                    //查看客户端是否退出                    if(QUIT.equals(msg)){                        System.out.println("客户端["+socket.getPort()+"]已退出");                        break;                    }                }            }        }catch (IOException e){            e.printStackTrace();        }finally {            if (serverSocket != null){                try {                    serverSocket.close();                    System.out.println("关闭ServerSocket");                } catch (IOException e) {                    e.printStackTrace();                }            }        }    }}

Client类:

package com.company.bio;

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

public class Client {    public static void main(String[] args)  {        final  String QUIT = "quit";        final String DEFAULT_SERVER_HOST = "127.0.0.1";        final int DEFAULT_SERVER_PORT = 8888;        Socket socket = null;        BufferedWriter bufferedWriter = null;

        try{            // 创建socket            socket = new Socket(DEFAULT_SERVER_HOST, DEFAULT_SERVER_PORT);            //创建IO流            BufferedReader bufferedReader = new BufferedReader(                new InputStreamReader(socket.getInputStream())            );            bufferedWriter = new BufferedWriter(                 new OutputStreamWriter(socket.getOutputStream())            );            //等待用户输入信息            BufferedReader consolReader = new BufferedReader(                    new InputStreamReader(System.in)            );            while (true) {                String input = consolReader.readLine();                //发送消息给服务器                bufferedWriter.write(input + "\n");                bufferedWriter.flush();                //读取服务器返回的消息                String msg = bufferedReader.readLine();                System.out.println(msg);

                //查看用户是否退出                if(QUIT.equals(input))break;            }        }catch (IOException e){            e.printStackTrace();        }finally {            if(bufferedWriter != null){                try {                    bufferedWriter.close();                    System.out.println("关闭socket");                } catch (IOException e) {                    e.printStackTrace();                }            }        }    }}

3.3 程序效果截图
Client:

在这里插入图片描述
Server:
在这里插入图片描述

更多资源和教程请关注公众号:非科班的科班。
如果觉得我写的还可以请给个赞,谢谢大家,你的鼓励是我创作的动力

基于BIO的实时Socket读写操作的更多相关文章

  1. 一个I/O线程可以并发处理N个客户端连接和读写操作 I/O复用模型 基于Buf操作NIO可以读取任意位置的数据 Channel中读取数据到Buffer中或将数据 Buffer 中写入到 Channel 事件驱动消息通知观察者模式

    Tomcat那些事儿 https://mp.weixin.qq.com/s?__biz=MzI3MTEwODc5Ng==&mid=2650860016&idx=2&sn=549 ...

  2. NX二次开发-基于MFC界面的NX对Excel读写操作(OLE方式(COM组件))

    NX二次开发API里没有对EXCAL读写操作的相关函数,市面上有很多种方法去实现,比如UFUN调KF,ODBC,OLE(COM组件)等等.这里我是用的OLE(COM组件)方式去做的,这种在VC上创建的 ...

  3. NX二次开发-基于NX开发向导模板的NX对Excel读写操作(OLE方式(COM组件))

    在看这个博客前,请读者先去完整看完:NX二次开发-基于MFC界面的NX对Excel读写操作(OLE方式(COM组件))https://ufun-nxopen.blog.csdn.net/article ...

  4. linux下mysql基于mycat做主从复制和读写分离之基础篇

    Linux下mysql基于mycat实现主从复制和读写分离1.基础设施 两台虚拟机:172.20.79.232(主) 172.20.79.233(从) 1.1软件设施 mysql5.6.39 , my ...

  5. 网络编程——基于TCP协议的Socket编程,基于UDP协议的Socket编程

    Socket编程 目前较为流行的网络编程模型是客户机/服务器通信模式 客户进程向服务器进程发出要求某种服务的请求,服务器进程响应该请求.如图所示,通常,一个服务器进程会同时为多个客户端进程服务,图中服 ...

  6. 基于AHB总线的master读写设计(Verilog)

    一.AHB总线学习 1. AHB总线结构 如图所示,AHB总线系统利用中央多路选择机制实现主机与从机的互联问题.从图中可以看出,AHB总线结构主要可分为三部分:主机.从机.控制部分.控制部分由仲裁器. ...

  7. socket读写返回值的处理

    在调用socket读写函数read(),write()时,都会有返回值.如果没有正确处理返回值,就可能引入一些问题 总结了以下几点 1当read()或者write()函数返回值大于0时,表示实际从缓冲 ...

  8. 基于Xenomai的实时Linux分析与研究

    转自:http://blog.csdn.net/cyberlabs/article/details/6967192 引 言 随着嵌入式设备的快速发展,嵌入式设备的功能和灵活性要求越来越高,很多嵌入式设 ...

  9. berkerly db 中简单的读写操作(有一些C的 还有一些C++的)

    最近在倒腾BDB,才发现自己确实在C++这一块能力很弱,看了一天的api文档,总算是把BDB的一些api之间的关系理清了,希望初学者要理清数据库基本知识中的环境,句柄,游标的基本概念,这样有助于你更好 ...

随机推荐

  1. vscode 添加golang插件

    安装好git 下列命令中的路径一定要按照自己实际的路径来 mkdir -p $GOPATH/src/golang.org/x  //路径下创建此文件cd $GOPATH/src/golang.org/ ...

  2. kafka性能测试代码

    bin/kafka-producer-perf-test.sh --num-records 5000000 --record-size 5000 \ --topic kafkatopic2 \ --b ...

  3. URL统一资源定位符

    URI 是统一资源标识符 URL 是统一资源定位符 ===================================================== 参考链接: 前端学HTTP之URL:ht ...

  4. JDBC 数据源

    概述 JNDI 数据源配置的相关内容已经在 JNDI 资源文档中详细介绍过.但从 Tomcat 用户的反馈意见来看,有些配置的细节问题非常棘手. 针对常用的数据库,我们已经给 Tomcat 用户提供了 ...

  5. mysql主从之binlog的工作模式

    一 三种模式介绍 1.1 查看mysql主库的binlog格式 binlog仅在主库设置即可,从库无需设置 binlog的默认方式为STATEMENT ( show variables like '% ...

  6. Redis事务、持久化、发布订阅

    一.Redis事物 1. 概念 Redis 事务可以一次执行多个命令, 并且带有以下两个重要的保证: 事务是一个单独的隔离操作:事务中的所有命令都会序列化.按顺序地执行.事务在执行的过程中,不会被其他 ...

  7. 初入webpack

    为什么需要构建工具? 通过caniuse我们了解到 现代浏览器对es6特性的支持程度: 由于现代浏览器对es6特性的支持度并不能说太高,为了兼容所以需要进行 es6语法的转换,除了此,三大框架的语法特 ...

  8. 全网最详细的Ceph14.2.5集群部署及配置文件详解,快来看看吧! -- <2>

    部署Ceph集群 Ceph版本选择 Ceph版本来源介绍 Ceph 社区最新版本是 14,而 Ceph 12 是市面用的最广的稳定版本. 第一个 Ceph 版本是 0.1 ,要回溯到 2008 年 1 ...

  9. 1064 朋友数 (20 分)C语言

    如果两个整数各位数字的和是一样的,则被称为是"朋友数",而那个公共的和就是它们的"朋友证号".例如 123 和 51 就是朋友数,因为 1+2+3 = 5+1 ...

  10. docker练习-堆栈

    介绍 分布式应用程序层次结构的顶部:堆栈. 堆栈是一组相互关联的服务,它们共享依赖关系,并且可以协调和缩放在一起. 单个堆栈能够定义和协调整个应用程序的功能(尽管非常复杂的应用程序可能希望使用多个堆栈 ...