Linux学习记录——사십 高级IO(1)
文章目录
- 1、IO
- 2、同、异步IO(5种IO类型)
- 3、其它高级IO
- 4、非阻塞IO
其它IO类型的实现在这篇之后的三篇
1、IO
input,output。调用read或recv接口时,如果对方长时间不向我方接收缓冲区拷贝数据,我们的进程就只能阻塞,这是读取条件不满足。阻塞的时间成本最后会体现在用户上。因此可以说,IO = 等 + 数据拷贝。高效IO则是单位事件内,等的比重越低,IO效率越高。
可以看出IO是有条件的,满足条件就叫IO时间就绪。
IO有五种模型
2、同、异步IO(5种IO类型)
调用读取用的接口后,对于有自己的接收缓冲区,有5种IO方式
阻塞IO:如果缓冲区内没有好数据,系统调用就一直阻塞等待,不做别的事情,直到准备好数据,才开始读取并返回。所有的套接字默认都是阻塞IO。
非阻塞IO:单次检验缓冲区内没有数据,recv函数就返回EWOULDBLOCK,每隔一段时间再调用recv接口来查看缓冲区,如果有就读取并返回数据,没有就还返回上面的。这也就是非阻塞轮询。
阻塞与非阻塞的区别在于等的方式不同,不过都要自己去拷贝数据。非阻塞在没得到结果前就不阻塞当前进程,直接返回,过一会再来查看;阻塞就是一直阻塞等待,直到读取数据再返回。
上面两个等的方式不同,但拷贝方式相同。
信号驱动IO:缓冲区内有数据时,就发送SIGIO信号通知上层调用接口来读取。信号是告知上层什么时候可以读取的,接口是在信号处理了之后才调用的。
recv函数调用后会先检测有没有数据,也就是条件检测,没有返回,有才读取再返回。像这样的接口只使用一个文件描述符。
多路复用/多路转接IO:执行一个进程,传多个文件描述符,调用多个接口来等待数据,进程只负责等待数据,等到有了数据,也就是IO条件就绪,就通知上层调用recv接口来读取并返回。
多路复用IO是效率最高的,因为并不只是一个进程在等,而是很多个进程在等,那么等待成功的可能性就更高。单个进程等待的概率和多个进程等待的概率相比。
上面4个都是同步IO。
异步IO:不参与等待过程,只是发起一个进程,给一个缓冲区来接收数据。至于结果怎么样,由操作系统来决定。多路复用是进程在等待,异步IO是系统在等待。
两种IO方式的区别是有没有参与等以及读取的过程。实际上,对应到系统,应当是一个进程通过特定的文件描述符,等待IO事件就绪,通过系统调用来读取数据拷贝到上层。
recv函数就是在做条件检测,条件通过再读取。
3、其它高级IO
记录锁、系统V流机制。存储映射IO(mmap),readv和writev函数等
4、非阻塞IO
int main()
{char buffer[64];while(true){printf(">>> ");fflush(stdout);ssize_t n = read(0, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n - 1] = 0;std::cout << "echo# " << buffer << std::endl;}}
}
这就是一个等待的过程,如果我们不输入,就一直卡在read那里。这是阻塞,如果要变成非阻塞,用fcntl接口。
#include <fcntl.h>
int fcntl(int fd, int cmd, …/* arg */);
cmd值有5种
复制一个现有的描述符(cmd=F_DUPFD)
获得/设置文件描述符标记(cmd=F_GETFD或F_SETFD)
获得/设置文件状态标记(cmd=F_GETFL或F_SETFL)//GET获得文件状态标记位,SET设置这个文件新的状态
获得/设置异步I/O所有权(cmd=F_GETOWN或F_SETOWN)
获得/设置记录锁(cmd=F_GETLK,F_SETLK或F_SETLKW)
void SetNonBlock(int fd) {int fl = fcntl(fd, F_GETFL);//使用F_GETFL将当前的文件描述符的属性取出来(这是一个位图)if (fl < 0) {perror("fcntl");return; }fcntl(fd, F_SETFL, fl | O_NONBLOCK);//然后再使用F_SETFL将文件描述符设置回去. 设置回去的同时, 加上一个O_NONBLOCK参数
}
非阻塞和阻塞都是文件的读取方式,在文件结构体中就是一个变量设为1或0,这里的O_NONBLOCK是宏,只有一个比特位,通过fcntl将这个数设置进文件结构体。
#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <cstdio>
#include <cstring>void SetNonBlock(int fd)
{int fl = fcntl(fd, F_GETFL);//使用F_GETFL将当前的文件描述符的属性取出来(这是一个位图)if (fl < 0){std::cerr << "error string: " << strerror(errno) << " error code: " << errno << std::endl;return ;}fcntl(fd, F_SETFL, fl | O_NONBLOCK);//然后再使用F_SETFL将文件描述符设置回去. 设置回去的同时, 加上一个O_NONBLOCK参数
}int main()
{char buffer[64];//SetNonBlock(0);//0表示非阻塞while(true){printf(">>> ");fflush(stdout);ssize_t n = read(0, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n - 1] = 0;std::cout << "echo# " << buffer << std::endl;}else if(n == 0){std::cout << "end file" << std::endl;break;} else{std::cout << "read error??" << std::endl;break;} }
}
不设置成0,就是阻塞。read那里不是0而是4,当前没有打开文件描述符为4的文件,就会read error。加上错误原因
else{std::cerr << "read error?" << "error string: " << strerror(errno) << " error code: " << errno << std::endl;break;}
现在再用上SetNonBlock,设置为0,那么即使不输入,也会出错,因为它不阻塞,直接去判断,没输入也没读取完,所以就打印错误语句
那么这里就在每一次循环后sleep一会。只有在输入时才会打印正常语句,不输入就一直打印错误语句。非阻塞IO在底层没有数据时,就以出错形式返回,不过不算正式地出错。
区分一下
else{if(errno == EAGAIN || errno == EWOULDBLOCK)//也就是错误码为11的两个,表示非阻塞式的出错返回{//底层数据没有准备好,下次继续检测sleep(1);std::cout << "data not ready" << std::endl;continue;}else if(errno == EINTR){//这次IO被信号中断,需要重新读取continue;}std::cerr << "read error? " << "error string: " << strerror(errno) << " error code: " << errno << std::endl;break;}
非阻塞IO在没有得到数据之前,可以做别的工作,模拟一下:
#include <iostream>
#include <unistd.h>
#include <fcntl.h>
#include <cstdio>
#include <cstring>
#include <vector>
#include <functional>void PrintLog()
{std::cout << "这个是一个日志例程" << std::endl;
}void OpenMysql()
{std::cout << "这个是一个操作数据库的例程" << std::endl;
}void CheckNet()
{std::cout << "这个是一个检测网络状态的例程" << std::endl;
}using func_t = std::function<void (void)>;
std::vector<func_t> funcs;void LoadTask()
{funcs.push_back(PrintLog);funcs.push_back(OpenMysql);funcs.push_back(CheckNet);
}void HandlerAllTask()
{for(const auto& func: funcs) func();
}void SetNonBlock(int fd)
{int fl = fcntl(fd, F_GETFL);//使用F_GETFL将当前的文件描述符的属性取出来(这是一个位图)if (fl < 0){std::cerr << "error string: " << strerror(errno) << " error code: " << errno << std::endl;return ;}fcntl(fd, F_SETFL, fl | O_NONBLOCK);//然后再使用F_SETFL将文件描述符设置回去. 设置回去的同时, 加上一个O_NONBLOCK参数
}int main()
{char buffer[64];SetNonBlock(0);//0表示非阻塞LoadTask();while(true){printf(">>> ");fflush(stdout);ssize_t n = read(0, buffer, sizeof(buffer) - 1);if(n > 0){buffer[n - 1] = 0;std::cout << "echo# " << buffer << std::endl;}else if(n == 0){std::cout << "end file" << std::endl;break;}else{if(errno == EAGAIN || errno == EWOULDBLOCK)//也就是错误码为11的两个,表示非阻塞式的出错返回{//底层数据没有准备好,下次继续检测HandlerAllTask();sleep(1);std::cout << "data not ready" << std::endl;continue;}else if(errno == EINTR){//这次IO被信号中断,需要重新读取continue;}std::cerr << "read error? " << "error string: " << strerror(errno) << " error code: " << errno << std::endl;break;}sleep(3); }
}
本篇gitee
结束。
相关文章:

Linux学习记录——사십 高级IO(1)
文章目录 1、IO2、同、异步IO(5种IO类型)3、其它高级IO4、非阻塞IO 其它IO类型的实现在这篇之后的三篇 1、IO input,output。调用read或recv接口时,如果对方长时间不向我方接收缓冲区拷贝数据,我们的进程就只能阻塞&a…...
【代码随想录】2
数组篇 二分查找 int search(int* nums, int numsSize, int target) { int left0; int rightnumsSize-1; while(left<right) {int mlddle(leftright)/2;if(nums[mlddle]>target){rightmlddle-1;}else if(nums[mlddle]<target){leftmlddle1;}else{return mlddle;}} r…...
TCP性能分析
ref: TCP性能和发送接收窗口、Buffer的关系 | plantegg...

RibbonGroup 添加QRadioButton
RibbonGroup添加QRadioButton: QRadioButton * pRadio new QRadioButton(tr("Radio")); pRadio->setToolTip(tr("Radio")); groupClipboard->addWidget(pRadio); connect(pRadio, SIGNAL(clicked(…...
一篇文章掌握WebService服务、工作原理、核心组件、主流框架
目录 1、WebService定义 解决问题: 2、WebService的工作原理 2.1 实现一个完整的Web服务包括以下步骤 2.2 调用方式 3、Web Service的核心组件 3.1 XML 3.2 SOAP 3.3 WSDL 3.4 UDDI 4、主流框架 4.1 AXIS(已淘汰) 4.2 XFire 4.3 CXF 5、Soap协议详解…...

观成科技-加密C2框架EvilOSX流量分析
工具简介 EvilOSX是一款开源的,由python编写专门为macOS系统设计的C2工具,该工具可以利用自身释放的木马来实现一系列集成功能,如键盘记录、文件捕获、浏览器历史记录爬取、截屏等。EvilOSX主要使用HTTP协议进行通信,通信内容为特…...

PCL 计算异面直线的距离
目录 一、算法原理二、代码实现三、结果展示四、相关链接本文由CSDN点云侠原创,PCL 计算异面直线的距离,爬虫自重。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 设置直线 A B AB A...

【数字人】9、DiffTalk | 使用扩散模型基于 audio-driven+对应人物视频 合成说话头(CVPR2023)
论文:DiffTalk: Crafting Diffusion Models for Generalized Audio-Driven Portraits Animation 代码:https://sstzal.github.io/DiffTalk/ 出处:CVPR2023 特点:需要音频对应人物的视频来合成新的说话头视频,嘴部抖…...

完成源示例
本主题演示如何创作和使用自己的完成源类,类似于 .NET 的 TaskCompletionSource。 completion_source 示例的源代码 下面的列表中的代码作为示例提供。 其目的是说明如何编写自己的版本。 例如,支持取消和错误传播不在此示例的范围内。 #include <w…...
业务和流程的关系
背景 概念不清,沟通就容易出现问题,最可怕会出现跑偏情况如何解决,数字化落地过程,程序是死的,最怕灵活,所以在沟通和编码,设计中,很重要的一点就是解决概念,澄清问题&a…...

【河海大学论文LaTeX+VSCode全指南】
河海大学论文LaTeXVSCode全指南 前言一、 LaTeX \LaTeX{} LATEX的安装二、VScode的安装三、VScode的配置四、验证五、优化 前言 LaTeX \LaTeX{} LATEX在论文写作方面具有传统Word无法比拟的优点,VScode作为一个轻量化的全功能文本编辑器,由于其极强的…...

学习python仅此一篇就够了(文件操作:读,写,追加)
python文件操作 文件编码 编码技术即:翻译的规则,记录了如何将内容翻译成二进制,以及如何将二进制翻译回可识别内容。 计算机中有许多可用编码: UTF-8 GBK BUG5 文件的读取操作 open()函数 在pyth…...
vue中 ref 和 $refs的使用
1. 作用 利用 ref 和 $refs 可以用于 获取 dom 元素, 或 组件实例 2. 获取 dom 使用步骤: 2.1 目标标签添加属性 :ref <div ref"chartRef">我是渲染图表的容器</div>2.2 通过$ref:获取标签 mounted() {console.log(this.$re…...

Centos7升级openssl到openssl1.1.1
Centos7升级openssl到openssl1.1.1 1、先查看openssl版本:openssl version 2、Centos7升级openssl到openssl1.1.1 升级步骤 #1、更新所有现有的软件包列表并安装最新的软件包: $sudo yum update #2、接下来,我们需要从源代码编译和构建OpenS…...

uniapp中实现H5录音和上传、实时语音识别(兼容App小程序)和波形可视化
文章目录 Recorder-UniCore插件特性集成到项目中调用录音上传录音ASR语音识别 在uniapp中使用Recorder-UniCore插件可以实现跨平台录音功能,uniapp自带的recorderManager接口不支持H5、录音格式和实时回调onFrameRecorded兼容性不好,用Recorder插件可避免…...

HashMap集合万字源码详解(面试常考)
文章目录 HashMap集合1.散列2.hashMap结构3.继承关系4.成员变量5.构造方法6.成员方法6.1增加方法6.2将链表转换为红黑树的treeifyBin方法6.3扩容方法_resize6.3.1扩容机制6.3.2源码resize方法的解读 6.4 删除方法(remove)6.5查找元素方法(get)6.6遍历HashMap集合几种方式 7.初始…...
LeetCode1124. Longest Well-Performing Interval
文章目录 一、题目二、题解 一、题目 We are given hours, a list of the number of hours worked per day for a given employee. A day is considered to be a tiring day if and only if the number of hours worked is (strictly) greater than 8. A well-performing in…...

如何使用手机公网远程访问本地群辉Video Station中视频文件【内网穿透】
最近,我发现了一个超级强大的人工智能学习网站。它以通俗易懂的方式呈现复杂的概念,而且内容风趣幽默。我觉得它对大家可能会有所帮助,所以我在此分享。点击这里跳转到网站。 文章目录 1.使用环境要求:2.下载群晖videostation&am…...

事件分析应急响应-Server2229(环境+解析)
任务环境说明: 服务器场景:Server2229(开放链接)用户名:root,密码:...

SpringCloud:微服务
文章目录 微服务服务架构演变单例架构(集中式架构)分布式架构 微服务SpringCloud 微服务 服务架构演变 单例架构(集中式架构) 单例架构: 将业务的所有功能集中在一个项目中开发,打成一个包部署 优点&…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
Axios请求超时重发机制
Axios 超时重新请求实现方案 在 Axios 中实现超时重新请求可以通过以下几种方式: 1. 使用拦截器实现自动重试 import axios from axios;// 创建axios实例 const instance axios.create();// 设置超时时间 instance.defaults.timeout 5000;// 最大重试次数 cons…...

2025季度云服务器排行榜
在全球云服务器市场,各厂商的排名和地位并非一成不变,而是由其独特的优势、战略布局和市场适应性共同决定的。以下是根据2025年市场趋势,对主要云服务器厂商在排行榜中占据重要位置的原因和优势进行深度分析: 一、全球“三巨头”…...
管理学院权限管理系统开发总结
文章目录 🎓 管理学院权限管理系统开发总结 - 现代化Web应用实践之路📝 项目概述🏗️ 技术架构设计后端技术栈前端技术栈 💡 核心功能特性1. 用户管理模块2. 权限管理系统3. 统计报表功能4. 用户体验优化 🗄️ 数据库设…...

用机器学习破解新能源领域的“弃风”难题
音乐发烧友深有体会,玩音乐的本质就是玩电网。火电声音偏暖,水电偏冷,风电偏空旷。至于太阳能发的电,则略显朦胧和单薄。 不知你是否有感觉,近两年家里的音响声音越来越冷,听起来越来越单薄? —…...
【生成模型】视频生成论文调研
工作清单 上游应用方向:控制、速度、时长、高动态、多主体驱动 类型工作基础模型WAN / WAN-VACE / HunyuanVideo控制条件轨迹控制ATI~镜头控制ReCamMaster~多主体驱动Phantom~音频驱动Let Them Talk: Audio-Driven Multi-Person Conversational Video Generation速…...
IP如何挑?2025年海外专线IP如何购买?
你花了时间和预算买了IP,结果IP质量不佳,项目效率低下不说,还可能带来莫名的网络问题,是不是太闹心了?尤其是在面对海外专线IP时,到底怎么才能买到适合自己的呢?所以,挑IP绝对是个技…...