springboot体会BIO(阻塞式IO)

news2025/1/12 12:24:05

使用springboot体会阻塞式IO

大致的思路为:
创建一个socket服务端,监听socket通道,并打印出socket通道中的内容。
创建两个socket客户端,向socket服务端写入消息。

1.创建服务端

public class RedisServer {
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(6379);

        while (true){
            System.out.println("redis服务端开始监听连接请求");
            //accept 阻塞式操作
            Socket socket = serverSocket.accept();
            System.out.println("redis监听到连接请求:" + UUID.randomUUID());
            InputStream inputStream = socket.getInputStream();
            int length;
            byte[] bytes = new byte[1024];
            //read 阻塞式操作
            while ((length = inputStream.read(bytes)) != -1){
                System.out.println("redis服务端成功读取请求体:"  + new String(bytes,0,length));
            }
        }
    }

}

2.创建客户端1

 public class RedisClient01 {

    public static void main(String[] args) throws IOException {
        Socket socket = new Socket("127.0.0.1",6379);
        OutputStream outputStream = socket.getOutputStream();
        while (true){
            Scanner scanner = new Scanner(System.in);
            String in = scanner.next();
            if("quit".equals(in)){
                break;
            }
            outputStream.write(in.getBytes());
            System.out.println("redis一号客户端完成写入操作");
        }
        outputStream.close();
        socket.close();
    }
}

3.创建客户端2

public class RedisClient02 {
    public static void main(String[] args) throws IOException {
        Socket socket = new Socket("127.0.0.1",6379);
        OutputStream outputStream = socket.getOutputStream();
        while (true){
            Scanner scanner = new Scanner(System.in);
            String in = scanner.next();
            if("quit".equals(in)){
                break;
            }
            outputStream.write(in.getBytes());
            System.out.println("redis二号客户端完成写入操作");
        }
        outputStream.close();
        socket.close();
    }
}

4.体验BIO(一)

我们先启动socket服务端
在这里插入图片描述
启动客户端1,再观察服务端的日志打印,成功监听客户端1与服务端建立了socket连接
在这里插入图片描述
我们使用客户端1写入数据,再观察服务端和客户端1的日志打印,客户端写入成功,服务端读取成功,一切安好
在这里插入图片描述
在这里插入图片描述
我们再启动第二个客户端,观察服务端的日志,没有任何新增的打印
在这里插入图片描述
难道是代码有问题?我们使用客户端2写入试试
在这里插入图片描述
写入是成功的,此时服务端的日志会有新增的打印吗?
在这里插入图片描述
依旧没有
当我把客户端1退出时,再来看一下服务端的日志
在这里插入图片描述
之前客户端2写入的消息成功被读取到了

5.小总结(一)

结合代码以及日志的打印结果分析:
服务端只能处理一个客户端的请求,第二个客户端发起请求,只能阻塞,直到第一个客户端的请求全部处理完毕。

6.如何处理阻塞问题?

我们借助多线程机制,使read方法“不阻塞”,修改我们服务端的代码

public class RedisServer {
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(6379);

        while (true){
            System.out.println("redis服务端开始监听连接请求");
            //accept 阻塞式操作
            Socket socket = serverSocket.accept();
            System.out.println("redis监听到连接请求:" + UUID.randomUUID());
            new Thread(() ->{
                try {
                    InputStream inputStream = socket.getInputStream();
                    int length;
                    byte[] bytes = new byte[1024];
                    //read 阻塞式操作
                    while ((length = inputStream.read(bytes)) != -1) {
                        System.out.println("redis服务端成功读取请求体:" + new String(bytes, 0, length));
                    }
                }catch (Exception e){
                    e.printStackTrace();
                }
            },Thread.currentThread().getName()).start();
        }
    }
}

7.体验BIO(二)

我们一样还是先启动服务端,再启动客户端1和客户端2,使用客户端1和客户端2向服务端写入消息,通过上述的例子我们得知,服务端会先处理客户端1的请求,客户端2的请求会阻塞。
服务端使用多线程之后,客户端2的请求就不会阻塞

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

8.小总结(二)

使用多线程后,确实能够处理多个客户端的请求,不会产生阻塞。但是来一个请求就new一个线程,对于操作系统来说是资源的严重浪费。

9.解决方案

(1)线程池
(2)NIO(非阻塞IO)

10.阻塞式IO模型
在这里插入图片描述

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

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

相关文章

Linux服务器应急响应(上)

目录 介绍步骤 介绍 网页篡改,即攻击者故意篡改网络上传送的报文,通常以入侵系统并篡改数据、劫持网络连接或插入数据等形式进行。 网页篡改一般有明显式和隐藏式两种。明显式网页篡改指攻击者为炫耀自己的技术技巧,或表明自己的观点实施的网…

《深入浅出WPF》读书笔记.11Template机制(下)

《深入浅出WPF》读书笔记.11Template机制(下) 背景 本文主要讲datatemplate和contenttemplate的联合使用,以及style的解析。 《深入浅出WPF》读书笔记.11Template机制(下) 代码 两者的作用域范围 datatemplate和contenttemplate的关系 两者的应用 指定目标类型…

Qt常用控件——QPushButton

QPushButton介绍 QWidget中涉及到的各种属性、函数、使用方法,对于Qt的各种控件都是有效的 使用QPushButton表示一个按钮,继承自QAbstracButton,这个类是一个抽象类 抽象类:包含纯虚函数,无法实例化出对象,…

第四届长城杯部分wp

还是太菜了,要经常练了 1.BrickGame 通过游戏就可以得到flag 2.SQLUP 一道文件上传的题目,在登陆页面我用admin和1登陆成功了,但是按照正常的应该是要爆破,用bp爆破得到下面的页面 登陆成功后,点击头像就可以进行文…

前端 PDF 预览技巧:标签 vs 插件,如何优雅地展示 PDF 文件

前言 pdf 作为一种常用的文档格式,相信很多同学都在项目中遇到过需要预览 pdf 文件的情况。其实实现的方式有很多,包括传统的标签 iframe 或 embed 方式,也可以运用一些插件,例如 pdf.js、vue-pdf 等等,本文将带大家一…

FastJson、Jackson、Gson、Hutool,JSON解析哪家强?JMH基准测试来排行

首发公众号:【赵侠客】 引言 在前面《释放你九成的带宽和内存:GZIP在解决Redis大Key方面的应用》一文中我使用GZIP算法对JSON格式数据进行了压缩,可以减小88%的空间,文中也提到了目前JSON格式在我们项目中应用的非常广泛几乎无处不在。压缩J…

整合Redis和RedisCacheManger

整合redis springboot在现在的版本中操作Redis数据库用到了lettuce,而不是Jedis,他们各有各的特点。Jedis以Redis命令作为方法名称,学习成本低,简单实用。但是Jedis实例是线程不安全的,多线程环境下需要基于连接池来使…

Java实现一个简单的本地群聊。可以多开Client。

网络编程也有趣的,Java中有对系统网络IO操作的封装包:Socket。现在我们在本地电脑(网络)用它来模拟一个简单的群聊功能,以便能更好地对网络编程进行深刻的理解。 "Client"去连接"Host",可同时多有…

JavaEE 第23节 TCP的流量控制与阻塞控制详解

目录 前言(必读)1、滑动窗口背景运行机制 2. 流量控制作用实现机制关键目标 3. 拥塞控制作用实现机制作用 4. 流量控制和拥塞控制的区别作用对象不同触发条件不同控制方式不同 5.总结 前言(必读) 流量控制(Flow Contr…

中科院院士薛其坤:通用量子计算机还得10-20年

说到量子计算机,很多人都抱有不切实际的幻想,甚至认为它无所不能,很快就能取代现有的电子计算机,但事实上,目前的量子计算机只能高效解决特定问题,不具备通用性。在2024年浦江创新论坛上,2023年…

linux日志备份

什么是日志文件?为什么要设立日志文件? 1、日志文件是用来记录事务对数据库的更新操作的文件。2、设立日志文件的目的是: 进行事务故障恢复;进行系统故障恢复;协助后备副本进行介质故障恢复。 但是,随着时间,日志文件内存过于增加,将会导…

creating chat agent with langchain and openai getting no attribute error

题意: 使用 LangChain 和 OpenAI 创建聊天代理时遇到“没有属性错误”(Getting "no attribute" error when creating a chat agent with LangChain and OpenAI) 问题背景: Im trying to test a chat agent using the …

基于springboot+vue的工作量统计系统(全套)

传统办法管理信息首先需要花费的时间比较多,其次数据出错率比较高,而且对错误的数据进行更改也比较困难,最后,检索数据费事费力。因此,在计算机上安装工作量统计系统软件来发挥其高效地信息处理的作用,可以…

java后端开发的DO、DTO、BO、AO、VO、POJO定义

1.常用文件夹命名规则 pojo: (1)vo (与前端交互的所有对象,包括接参和返回) (2)query (查询的筛选条件,前端传参和后端内部传参通用) &#x…

[计算机基础四大件学习笔记]计算机组成原理

文章总览:YuanDaiMa2048博客文章总览 计算机基础四大件学习笔记 说明:虽然学习过计算机组成与系统结构、操作系统、计算机网络以及数据结构这四门课程,但是对于这四门课的结合和实际中的应用还是模糊的,因此x想通过网上的一些视频…

Rust语言初探:WebAssembly 入门

Rust语言初探:WebAssembly 入门 前言 在我的印象中,Rust 一直是比较底层的语言,例如在操作系统底层、高性能中间件等底层场景才会看到它的身影。 然而,随着技术的发展,Rust 也开始在前端场景如 WebAssembly 中崭露头…

大数据Flink(一百一十五):Flink SQL的基本概念

文章目录 Flink SQL的基本概念 一、​​​​​​​SQL 中表的概念 二、​​​​​​​​​​​​​​SQL 临时表、永久表 三、​​​​​​​​​​​​​​SQL表类型的定义 四、​​​​​​​​​​​​​​常见的连接器 五、​​​​​​SQL数据视图 1、​​​​​​…

校园气膜馆助力青少年体质发展:少年强则国强—轻空间

青少年是国家的未来,体质的强健与否,直接关系到国家的竞争力和可持续发展。在现代社会,学习和压力并存,青少年的体育锻炼时间不断被压缩,如何提供更为优质的体育设施,帮助他们增强体质,成为学校…

【C/C++】“秒懂”学C/C++不可错过的“经典编程题” — 日期类的经典运用 (含题链接)

“秒懂”学C/C不可错过的“经典编程题” — 日期类的经典运用 (含题链接) 1. 计算日期到天数转换(1). 解题思路:(2). 代码实现: 2. 打印日期(1). 解题思路:(2). 代码实现: 3. 日期累加(1). 解题思路:(2). 代…

Java 设计模式-状态模式

目录 一. 概述 二. 主要角色 三. 代码示例 四. 优缺点 优点: 缺点: 五. 常见应用场景 一. 概述 状态模式是一种行为设计模式,它允许一个对象在其内部状态改变时改变它的行为。对象看起来好像修改了它的类。状态模式把所有的与一个特定…