网络编程套接字(4)——Java套接字(TCP协议)

目录

一、Java流套接字通信模型

二、TCP流套接字编程 

1、ServerSocket

ServerSocket构造方法:

ServerSocket方法:

2、Socket

Socket构造方法:

Socket方法:

三、代码示例:回显服务器

1、服务器代码

代码解析

2、客户端代码

代码解析

3、注意事项

        (1)缓冲区

        (2)socket的close,释放文件描述符表

        (3)多线程的应用

        (4)引入线程池的改进

                1、协程

                2、IO多路复用

4、执行代码

        前述:

5、客户端和服务器交互的过程


一、Java流套接字通信模型

        

        1.客户端和服务端:开发时,经常是基于一个主机开启两个进程作为客户端和服务端,但是真实的场景,一般是不同主机。

        2.注意目的IP和目的端口号,标识了一次数据传输时要发送数据的终点主机和进程。

        3.Socket编程我们是使用流套接字和数据报套接字,基于TCP或UDP协议,但应用层协议,也需要考虑。


二、TCP流套接字编程 

        TCP面向字节流,和UDP面向数据报不同,但是写的回显服务器中心思想是一样的,代码会有不同。以下API介绍。

1、ServerSocket

        这个Socket类对应到网卡,只能给服务器使用。

        ServerSocket是创建TCP服务端Socket的API。

ServerSocket构造方法:

方法签名方法说明
ServerSocket(int port)创建一个服务端流套接字Socket,并绑定到指定端口

ServerSocket方法:

方法签名方法说明
Socket accept()

开始监听指定端口(创建时绑定的端口),有客户端

连接后,返回一个服务端socket对象,并基于该

Socket建立与客户端的连接,否则阻塞等待

void close()关闭此套接字

2、Socket

        对应到网卡,既可以给客户端使用,也可以给服务器使用。

        Socket是客户端Socket,或服务端接收到客户端建立的连接(accept方法)的请求后,返回的服务端Socket。

        不管是客户端还是服务端Socket,都是双方建立连接之后,保存对端信息,及用来与对方收发数据的。

Socket构造方法:

方法签名方法说明
Socket(String host, int port)

创建一个客户端流套接字Socket,并与对应IP的主机

上,对应端口的进程进行连接

Socket方法:

方法签名方法说明
InetAddress getInetAddress()返回套接字所连接的地址
InputStream getInputStream()返回此套接字的输入流
OutputStream getOutputStream()返回此套接字的输出流

三、代码示例:回显服务器

1、服务器代码

public class TcpEchoServer {ServerSocket socket = null;public TcpEchoServer(int serverPort) throws IOException {socket = new ServerSocket(serverPort);}public void start() throws IOException {System.out.println("服务器启动");ExecutorService pool = Executors.newCachedThreadPool();while (true) {//通过 accept 这个方法来 “接听电话”,然后才能通信Socket clientSocket = socket.accept();
//            Thread t = new Thread(() -> {
//                //通过这个方法来处理一次连接,连接的过程会涉及到多次请求响应交互
//                processConnection(clientSocket);
//            });
//            t.start();pool.submit(new Runnable() {@Overridepublic void run() {processConnection(clientSocket);}});}}//通过这个方法来处理一次连接,连接的过程会涉及到多次请求响应交互private void processConnection(Socket clientSocket) {System.out.printf("[%s : %d] 客户端上线\n", clientSocket.getInetAddress(), clientSocket.getPort());//循环读取客户端请求并返回响应try(InputStream inputStream = clientSocket.getInputStream();OutputStream outputStream = clientSocket.getOutputStream()) {while (true) {Scanner scanner = new Scanner(inputStream);if(!scanner.hasNext()) {//读取完毕,客户端断开连接,就会产生读取完毕System.out.printf("[%s : %d] 客户端下线\n", clientSocket.getInetAddress(), clientSocket.getPort());break;}//1、接受从客户端发来的请求,解析请求(将请求转换为字符串)为了方便,直接使用Scanner读取//  读取请求并解析. 这里注意隐藏的约定. next 读的时候要读到空白符才会结束.//    因此就要求客户端发来的请求必须带有空白符结尾. 比如 \n 或者空格.//客户端发来的请求要包含 “\n”String request = scanner.next();//2、计算请求String response = process(request);//3、把计算的响应返回给客户端//也可以通过下面的这种方式写回,但下面这种方式不好添加 "\n"//outputStream.write(response.getBytes(), 0, response.getBytes().length);// 也可以给outputStream套上一层,可以更方便的加上 "\n"PrintWriter writer = new PrintWriter(outputStream);writer.println(response);//刷新缓冲器writer.flush();//打印日志System.out.printf("[%s : %d] res: %s resp: %s\n", clientSocket.getInetAddress(),clientSocket.getPort(), request, response);}} catch (IOException e) {throw new RuntimeException(e);} finally {try {clientSocket.close();} catch (IOException e) {throw new RuntimeException(e);}}}private String process(String request) {return request;}public static void main(String[] args) throws IOException {TcpEchoServer tcpEchoServer = new TcpEchoServer(9090);tcpEchoServer.start();}
}

代码解析

        服务器通过accept方法,和客户端建立联系

        如上图,应用程序代码中调用对应的 api 和服务器尝试建立连接,内核就会发起连接的流程。

        服务器的内核就会配合客户端这边的工作来完成连接的建立。

        这个连接建立的过程,就相当于:电话这边在拨号,另一边在响铃;但是要等到用户点击了接听,然后才能进行后续的通信。

        内核建立的连接不上决定性的,还需要用户程序把这个连接进行 "接听" / accept 操作,然后才能进行后续的通信。

        注意:accept也是一个可能会产生阻塞的操作,如果当前没有客户端连过来,此时 accept 就会阻塞。

        有一个客户端连过来,accept 一次就能返回一次。

        有若干个客户端连过来,accept 就需要执行多次。

        第一个socket是负责客户端的连接第二个clientSocket是负责操作服务器内部的业务。

        接下来的方法是处理连接的交互,新创建多出来线程后面讲

        TCP是有连接的和面向字节流的,从下面代码就可以看出来

        TCP的 socket 是可以保存对端的信息InputStream是从网卡读数据,outputStream是从网卡写数据。

        TCP面向字节流,这里的字节流和 文件 中的字节流完全一样,使用文件操作一样的方法和类的对 TCP 的 socket 进行读写。

        如图:

此处的读操作完全可以通过 read 来完成,read 是把收到的数据放到 byte 数组中,后续根据请求处理响应还需要把这个 byte 数组转成 String,比较麻烦,还有一个更简单的方法:Scanner,如上图。

        如图:

        客户端退出之后,服务器就能感知到 “客户端下线” 的操作客户端退出的时候,就会触发TCP的“断开连接”流程,服务器这边的代码就能感知到,对应的Scanner就能够在hasNext这里返回false。

        这里要用 scanner.next() ,因为接受来的请求的字节流,要知道什么时候结束发送过来的请求会带有 "\n",发来的请求中有空白符,比如 \n 或 空格。

        接下来是计算请求,如图:

        因为这里是简单的回显服务器,所以计算就直接返回请求的内容,process方法如图:

        返回响应

        记得要刷新缓冲器。

        

2、客户端代码

public class TchEchoClient {Socket socket = null;public TchEchoClient(String serverIp, int serverPort) throws IOException {//这里的ip和port是直接发给socket的对象// 因为TCP是有连接的,所以socket会保存ip和port这些信息//因此TcpEchoClient不必保存ip和portsocket = new Socket(serverIp, serverPort);}public void start() {System.out.println("客户端启动");try(InputStream inputStream = socket.getInputStream();OutputStream outputStream = socket.getOutputStream();Scanner scannerConsole = new Scanner(System.in);Scanner scannerNetwork = new Scanner(inputStream);PrintWriter writer = new PrintWriter(outputStream)) {while (true) {System.out.print("->");//1、从控制台输入请求,构造请求if(!scannerConsole.hasNext()) {break;}//发送给服务器的字符串要带有 "\n"String request = scannerConsole.next();//2、把请求发给发送给服务器,这里需要用println来发生,确保信息里面有 "\n"//这里是和服务器的scanner.next()对应的writer.println(request);//通过flush刷新缓冲器,确保数据真的发出去了writer.flush();//3、接收服务器返回的响应//这里也是和服务器返回的响应逻辑对应,返回的响应带有 "\n"String response = scannerNetwork.next();//4、把返回的响应显示到控制台System.out.println(response);}} catch (IOException e) {throw new RuntimeException(e);}}public static void main(String[] args) throws IOException {TchEchoClient tchEchoClient = new TchEchoClient("127.0.0.1", 9090);tchEchoClient.start();}
}

代码解析

        构造方法里面,如图:

        执行客户端里的构造方法,就会和对应的服务器进行TCP的连接建立流程。(系统内核完成的)。

        这边把内核中连接的流程走完了,服务器这边就能够从 accept 返回

        服务器的建立连接,如图

        processConnection方法内部

        然后进入while循环,执行服务器的内部逻辑,如图:

3、注意事项

        (1)缓冲区

        文件的IO操作都是比较低效的,所以就希望能够让低效的操作,进行的尽量少一些。

        解决方案:引入缓冲区(内存),先把写入网卡的数据放到内存缓冲区中,等攒一波再统一进行发送(把多次IO合并成一次)

        但也有个问题,如果发送的数据很少。此时由于缓冲区还没满,数据就待在缓冲区里,没有被真正发送出去。所以上面的代码中要加入刷新缓冲区的代码。(不然就连数据都发送不出去),如图:不管数据有没有真的发送出去,都要进行刷新缓冲器

        (2)socket的close,释放文件描述符表

        如图:

        clientSocket对象要释放掉因为客户端不止一个,会有很多个,一个客户端发来请求就会占用一个文件描述符表,我们不确定客户端什么时候下线,就可能会一直占用着这些资源(文件描述符表),随着客户端越来越多,又无法释放,文件描述符表就可能占满

        而服务器的sock就不用释放,调用close方法,如图:

        因为这个socket会伴随着服务器的生命周期很长,整个生命周期都会使用到它,而且也只有一个就不用担心占不占满文件描述符表了,像这种情况就不用释放;只要程序退出,socket也会随着进程的销毁一起被释放;UDP的回显服务器的socket不用释放也是因为这个原因。

        因为有finally最后会执行socket的释放,所以,释放了 socket 对象,上述流对象不释放,也问题不大。这两流对象内部不持有文件描述符,只是只有一些内存结构。内存结构可以被 gc 释放。

        但是只释放了流对象,不释放socket,就不行了socket持有了文件描述符表,本质还是要释放文件描述符资源。

        不过这里使用try with resources 的版本,也给它close了,更保险一点

        (3)多线程的应用

        服务器支持多个客户端同时访问是天经地义的,但如果不加多线程方案执 processConnection方法,如图:

        当有多个客户端想同时访问时,第一个客户端先访问服务器,服务器就会从accept这返回(解除阻塞),进入到processConnection中了,接下来就会在scanner.hasNext返回,继续执行服务器逻辑,因为有while循环,完成服务器的逻辑后,把响应返回给客户端执行完上述一轮操作后,循环回来继续再hasNext阻塞,等待下一次循环,知道客户端退出,连接结束,服务器中的循环才会结束、退出,如图:

        当有第二个客户端想访问服务器时,因为第一个客户端还没执行完服务器还在里面的while循环转圈圈呢就无法第二次执行到accept

        这里虽然第二个客户端和服务器在内核层面上建立了TCP连接,但是应用程序这里,无法把连接拿到的应用程序,在服务器程序里面进行处理(像是别人给你打电话,你手机一直在响,但是你没接)。

        如果第一个客户端退出了,第二个客户端之前的请求为啥就会被立即出来,而没有丢弃呢?这是因为当前TCP在内核中,每个 socket 都是有缓冲区的客户端发送的数据确实是发了,服务器也收到了,只不过数据是在服务器的接受缓冲区中。

        一旦第一个客户端退出了,回到第一层循环,继续执行第二次 accept ,继续执行 next 就能把之前缓冲区的内容给读出来像是菜鸟驿站,可以存放快递包裹)。

        单个线程,无法既能给客户端提供循环提供服务,又能快速的调用到第二次accept

        所以这里的核心思路就是使用多线程,也是简单的办法,引入多线程,主线程负责执行 accept每次有一个客户端连上来,就分配一个新的线程,由新的线程负责给客户端提供服务。如图:

        而上述没有引用多线程而造成的问题,并不是 TCP 的问题,而是代码本身的问题,因为两层循环嵌套而导致的问题。UDP只有一层循环,所以就不涉及到这种问题,之前的UDP天然的就能处理多个客户端的请求。

        (4)引入线程池的改进

        如图:

        这里每次来一个客户端,就会创建一个新的线程;每次这个客户端结束,就要销毁这个线程。如果客户端比较多,就会使服务器 频繁创建、销毁 线程

        因此,这里我们可以引入线程池,代码如图:

        线程池,解决的事频繁创建销毁的问题。

        如果当前场景是线程频繁创建,但是不销毁呢?(扩展话题)

        每个客户端如果处理过程都很短(网站),线程池可以解决这种频繁创建消耗的问题

        但是每个客户端处理过程都很长呢(例如吃鸡、王者、LOL等待),如果继续使用线程池 / 多线程,此时就会导致当前的服务器上一下积累了大量线程,此时对于服务器的负担就会非常重!!

        为了解决上述积累大量线程的问题,可以引入以下的方案:

                1、协程

                轻量级线程。本质还是一个线程,用户态可以通过手动调度的方式,让这一个线程 “并发” 的做多个任务。(Go / Python)

                2、IO多路复用

                系统内核级别的机制本质上是让一个线程同时去负责处理多个 socket本质在于这些 socket 数据并非是同一时刻都需要处理。

                基本盘在于,虽然有多个 socket ,但是同一时刻活跃的 socket 只是少数(需要读写数据的 socket),大部分 socket 都是在等,使用一个线程来等多个 socket。        就像去路边摊吃小吃,有很多小吃,我可以在我想吃的小吃店,依次都付款,然后站在这些店的中间,哪个路边摊先做好就去哪个路边摊拿小吃,这样等的过程,就可以节约出很多时间。

4、执行代码

        前述:

        要想执行多个客户端程序,我们要设置一些东西,设置方法如图:

        执行代码后,服务器和客户端交互,如图:

        客户端:

        服务器:

        多个客户端发送数据给服务器:

        可以看到,不同的客户端的进程号不同。而且服务器可以同时被多个客户端请求。

5、客户端和服务器交互的过程

1、服务器启动,阻塞在 accept,等待客户端发来的请求

2、客户端启动

        这里的 new 操作,触发了和服务器之间的建立连接的操作,此时 服务器 就会从 accept 中返回。

 3、服务器从 accept 返回,进入到 processConnection方法中

        执行到 hasNext 这里,产生阻塞,此时虽然连接建立了,但客户端还没发来任何请求,hasNext 阻塞等待请求到达。类似电话通了,但没人说。

4、客户端继续执行到 hasNext,等待用户从控制台写入信息。

5、用户真的输入了,此时 hasNext 就返回了,继续执行这里的发送请求的逻辑。

6、服务器从 hasNext 返回读取到请求内容,并进行处理

        读取到请求,构造成响应,并把响应返回给客户端

        服务器就结束本次循环,开启下一轮循环,继续阻塞在 hasNext 等待下一个请求。

7、客户端读到响应,并且显示出来

        结束本次循环,进入下一次循环,继续阻塞等待在 hasNext 等待用户下一次的输入。


都看到这了,点个赞再走吧,谢谢谢谢谢

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/745271.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

配置阿里云加速器

国内镜像中心常用阿里云或者网易云。在本地docker中指定要使用国内加速器的地址后&#xff0c;就可以直接从阿里云镜像中心下载镜像。 2024阿里云-上云采购季-阿里云 [rootlocalhost /]# mkdir -p /etc/docker [rootlocalhost /]# tee /etc/docker/daemon.json <<-EOF &…

windows 安装 gitlab-runner CICD

点击搜索图标 手动输入PowerShell, 右键点击管理员权限打开&#xff0c; 一、安装 安装 gitlab runner 文档参考地址 1、下载exe执行文件 我这里是 win64 https://gitlab-runner-downloads.s3.amazonaws.com/latest/binaries/gitlab-runner-windows-amd64.exe 2、创建 gitla…

论文研读笔记2

1.Han异构图网络看不懂。。。正在看 https://zhuanlan.zhihu.com/p/346658317 https://www.bilibili.com/video/BV1fc411z7mx?p19&vd_source6fb54905ed1c5c332b7a606643d8525c 2. 阅读论文&#xff1a;Learning Long- and Short-Term User Literal-Preference with Mu…

Java日志框架Log4j 2详解

有时希望能够以文件的形式记录执行过程中出现的异常信息&#xff0c;甚至记录程序正常运行的关键步骤&#xff0c;以便日后查看&#xff0c;那么该如何处理呢? 答:显然&#xff0c;可以自行编程实现这个需求&#xff0c;但是&#xff0c;从更注重效率和性能的方面考虑&#x…

物联网终端telegraf采集设备信息

背景 低功耗设备上资源有限&#xff0c;但又比较重要。对其的管理难度很大&#xff0c;有些时候又必须时刻了解其运行状况。我们自然想到的是能否有办法监控它呢&#xff1f;当时是有的&#xff01;而且很成熟的解决方案。TICK技术栈&#xff0c;那TICK是什么呢&#xff1f; TI…

【剪枝实战】使用VGGNet训练、稀疏训练、剪枝、微调等,剪枝出只有3M的模型

摘要 本次剪枝实战是基于下面这篇论文去复现的&#xff0c;主要是实现对BN层的γ/gamma进行剪枝操作&#xff0c;本文用到的代码和数据集都可以在我的资源中免费下载到。 相关论文&#xff1a;Learning Efficient Convolutional Networks through Network Slimming (ICCV 2017…

Ubuntu18.04下opencv基础操作(打开图片及视频)

文章目录 一、认识opencv一、编写一个打开图片进行特效显示的代码二、使用opencv库编写打开摄像头压缩视频的程序2.1 虚拟机获取摄像头权限2.2 播放视频2.3 录制视频 三、总结四、参考资料 一、认识opencv 开源计算机视觉(OpenCV)是一个主要针对实时计算机视觉的编程函数库。 …

WanAndroid(鸿蒙版)开发的第四篇

前言 DevEco Studio版本&#xff1a;4.0.0.600 WanAndroid的API链接&#xff1a;玩Android 开放API-玩Android - wanandroid.com 其他篇文章参考&#xff1a; 1、WanAndroid(鸿蒙版)开发的第一篇 2、WanAndroid(鸿蒙版)开发的第二篇 3、WanAndroid(鸿蒙版)开发的第三篇 …

Gitee 服务器

Git 服务器集成 1. 创建仓库 2. 远程仓库简易操作指令 # Git 全局设置&#xff0c;修改成自己的信息 git config --global user.name "Muko" git config --global user.email "txk0x7d2163.com" # 创建 git 仓库&#xff0c;基本操作指令和其他远程仓库一…

Ps 滤镜:中间值

Ps菜单&#xff1a;滤镜/杂色/中间值 Filter/Noise/Median 中间值 Median滤镜可用于减少或消除图像中的噪点和杂色&#xff0c;同时能较好地保留图像边缘和细节信息。 中间值滤镜通过计算一个像素周围一定区域内的像素值的中间值&#xff08;即这些值的中位数&#xff09;&…

群集----Memcached

一、NoSQL介绍 NoSQL是对 Not Only SQL、非传统关系型数据库的统称。 NoSQL一词诞生于1998年&#xff0c;2009年这个词汇被再次提出指非关系型、分布式、不提供ACID的数据库设计模式。 随着互联网时代的到来&#xff0c;数据爆发式增长&#xff0c;数据库技术发展日新月异&a…

怎么避免电脑数据被拷贝?电脑如何禁用USB功能?

在无纸化办公的今天&#xff0c;很多重要数据都存放在电脑中。为了避免数据泄露&#xff0c;需要采用安全的方式保护电脑数据。那么&#xff0c;该如何避免电脑数据被拷贝呢&#xff1f;下面我们就来了解一下。 方法一&#xff1a;物理隔绝 物理隔绝是一种原始但有效的USB禁用…

《小程序从入门到入坑》框架语法

前言 哈喽大家好&#xff0c;我是 SuperYing&#xff0c;我们继续小程序入门系列&#xff0c;本文将对小程序框架语法进行比较全面的介绍。在《小程序从入门到入坑》简介及工程创建中&#xff0c;我们提到小程序项目结构&#xff0c;主要包括 app.json&#xff0c;app.js&…

STM32初识1

什么是单片机&#xff1f; 单片机&#xff08; Single-Chip Microcomputer &#xff09;是一种集成电路芯片&#xff0c;把具有数据处理能力的中央处 理器 CPU 、随机存储器 RAM 、只读存储器 ROM 、多种 I/O 口和中断系统、定时器 / 计数器等功 能&#xff08;可能还包括显示…

部署快捷、使用简单、推理高效!大模型部署和推理框架 Xinference 来了!

今天为大家介绍一款大语言模型&#xff08;LLM&#xff09;部署和推理工具——Xinference[1]&#xff0c;其特点是部署快捷、使用简单、推理高效&#xff0c;并且支持多种形式的开源模型&#xff0c;还提供了 WebGUI 界面和 API 接口&#xff0c;方便用户进行模型部署和推理。 …

【iOS】ARC学习

文章目录 前言一、autorelease实现二、苹果的实现三、内存管理的思考方式__strong修饰符取得非自己生成并持有的对象__strong 修饰符的变量之间可以相互赋值类的成员变量也可以使用strong修饰 __weak修饰符循环引用 __unsafe_unretained修饰符什么时候使用__unsafe_unretained …

蓝桥杯--冶炼金属

目录 一、题目 二、解决代码 &#xff08;1&#xff09;版本一&#xff08;报错&#xff1a;超时&#xff09; 代码分析 &#xff08;2&#xff09;版本二&#xff08;不会超时&#xff09; 代码分析 &#xff08;3&#xff09;版本三&#xff08;最终精简版&#xff09;…

Python自学☞序列和索引的相关操作

一、基本概念 1、概念 序列是一个用于存储多个值的连续空间&#xff0c;每个值都对应一个整数的编号&#xff0c;称为索引 2、切片的语法结构 注&#xff1a;切片可以访问序列一定范围内的元素 序列[start&#xff1a;end&#xff1a;step] start-->切片的开始索…

Python数据分析-5

1.时间序列 2.pandas重采样 重采样&#xff1a;指的是将时间序列从一个频率转化为另一个频率进行处理的过程&#xff0c;将高频率数据转化为低频率数据为降采样&#xff0c;低频率转 化为高频率为升采样。 统计出911数据中不同月份电话次数的变化情况&#xff1a…

vue3中的文字滚动播报

vue3中的文字滚动播报 之前UI框架一直使用的elementPlus&#xff0c;有个需求&#xff0c;需要在页面上写个滚动播放新闻的功能&#xff0c;发现UI框架居然没有这个组件。花了一下午&#xff0c;在ChatGPT的帮助下&#xff0c;总算写成功了&#xff0c;先看最终展示效果 web页…