Linux 【C编程】IO进阶— 阻塞IO、非阻塞IO、 多路复用IO、 异步IO

文章目录

  • 1.阻塞IO与非阻塞IO
    • 1.1为什么有阻塞式?
    • 1.2非阻塞
  • 2.阻塞式IO的困境
  • 3.并发IO的解决方案
    • 3.1非阻塞式IO
    • 3.2多路复用IO
      • 3.2.1什么是多路复用IO?
      • 3.2.1多路复用IO select原理
      • 3.2.1多路复用IO poll原理
    • 3.3异步IO

1.阻塞IO与非阻塞IO

1.1为什么有阻塞式?

1.常见的阻塞:wait pause sleep 等函数 ; read或write某些文件
2.阻塞式的好处:在某些情况下,阻塞式 I/O 可以更有效地利用系统资源。在一些高负载场景下,阻塞式 I/O 可以避免频繁的上下文切换,降低系统开销。 阻塞式 I/O 的编程模型通常比较直观和易于理解。代码顺序执行,不需要太多复杂的逻辑来处理异步操作和事件回调。

1.2非阻塞

1.为什么要实现非阻塞?
非阻塞 I/O 允许程序在进行 I/O 操作时不被阻塞,可以继续执行其他任务。这对于高并发的应用程序尤其重要,可以充分利用系统资源,提高系统的吞吐量和性能。 在一些场景下,程序需要快速响应并处理多个客户端请求。如果使用阻塞 I/O,一个慢速的 I/O 操作可能会导致整个程序被阻塞,无法及时响应其他请求,而非阻塞 I/O 可以避免这种情况。非阻塞 I/O 可以与异步 I/O 结合,使得程序可以发起一个 I/O 操作后继续执行其他任务,当 I/O 完成时,系统通知程序并处理完成的数据。这种模式可以提高系统的并发性和性能。
2.如何实现非阻塞IO访问: O_NONBLOCK和fcntl

2.阻塞式IO的困境

以在程序中读取键盘为例

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h>
int main(){
    char buf[199];
    memset(buf,0,sizeof(buf));
    //读取键盘
    // 键盘就是标准输入, stdin
    printf("before read.\n");
    read (0,buf,2);  //从键盘读取两个字节 就是0号文件标识符 read 本身就是阻塞式的
    printf("读出来的内容是 :【%s】\n",buf);
    return 0;
}

此时运行程序后,发现已经堵塞住了 正在等待输入
在这里插入图片描述
在程序中读取鼠标
鼠标设备本质上也是字符设备 在/dev/input中,使用cat 读取 mouse 可能有多个 哪个晃动鼠标能得到数据就说明是鼠标设备,至于乱码是因为读取的是二进制数据,不是普通的字符。
在这里插入图片描述
代码演示:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h>
#include <fcntl.h>
int main(){
    char buf[199];
    memset(buf,0,sizeof(buf));
    //读取鼠标
    // 鼠标不是标准输入 需要open 打开
    int fd = -1;
    fd = open("/dev/input/mouse0",O_RDONLY);
    if(fd<0){
        perror("open:");
        return -1;
    }

    printf("before read mouse.\n");
    read (fd,buf,2);  //从鼠标读取两位
    printf("mouse读出来的内容是 :【%s】\n",buf);




    return 0;
}

如果程序同时读取键盘和鼠标

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h>
#include <fcntl.h>
int main(){
    char buf[199];
    memset(buf,0,sizeof(buf));
    //读取鼠标
    // 鼠标不是标准输入 需要open 打开
    int fd = -1;
    fd = open("/dev/input/mouse0",O_RDONLY);
    if(fd<0){
        perror("open:");
        return -1;
    }

    printf("before read mouse.\n");
    read (fd,buf,2);  //从鼠标读取两位
    printf("mouse读出来的内容是 :【%s】\n",buf);

    memset(buf,0,sizeof(buf));
    printf("before keyboard read.\n");
    read (0,buf,2);  //从键盘读取两个字节 就是0号文件标识符 read 本身就是阻塞式的
    printf("keyboard 读出来的内容是 :【%s】\n",buf);


    return 0;
}

如果用户使用这个程序,先使用鼠标,再使用键盘输入,程序没有问题
在这里插入图片描述
当用户先输入键盘的时候,阻塞IO就出现问题了,程序必须先等待鼠标事件,当前程序已经被阻塞住了,无论怎么输入键盘,程序也不会有响应。例如下图
在这里插入图片描述

3.并发IO的解决方案

3.1非阻塞式IO

使用fcntl 修改0号文件标识符的属性,添加非阻塞属性。
鼠标是通过open 打开的,直接添加非阻塞属性即可。

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h>
#include <fcntl.h>
int main(){
    char buf[199];
    memset(buf,0,sizeof(buf));
    int ret = -1;
    // 鼠标不是标准输入 需要open 打开
    int fd = -1;
    fd = open("/dev/input/mouse0",O_RDONLY|O_NONBLOCK);  //添加非阻塞属性
    if(fd<0){
        perror("open:");
        return -1;
    }
    int flag=  -1;
    //把0号描述符变成非阻塞式的
    flag = fcntl(0,F_GETFL); //获取原来的flag
    flag |= O_NONBLOCK ; //添加非阻塞属性
    fcntl(0,F_SETFL,flag); //更新flag
    while (1){
        ret  = read (fd,buf,2);  //从鼠标读取
        if(ret>0){
            printf("mouse读出来的内容是 :【%s】\n",buf);
            memset(buf,0,sizeof(buf));
        }
        ret  = read (0,buf,10);  //从键盘读取 
        if(ret>0){
            printf("keyboard 读出来的内容是 :【%s】\n",buf);
            memset(buf,0,sizeof(buf));
        }
    }
   


    return 0;
}

与阻塞相比,想读入键盘就读入键盘,想读入鼠标就读入鼠标,提高并发性
在这里插入图片描述

3.2多路复用IO

3.2.1什么是多路复用IO?

多路复用 I/O 是一种机制,允许一个进程能够同时监视和处理多个 I/O 源,例如文件描述符、sockets 或其他文件 I/O。这些多路复用的系统调用允许程序等待多个 I/O 事件中的任何一个就绪,从而避免了阻塞等待单个 I/O 完成的情况,提高了程序的效率和并发处理能力。对外部还是阻塞式的,内部非阻塞式自动轮询多路IO看看哪个有数据,就先用哪个。

3.2.1多路复用IO select原理

在Linux系统上常见的多路复用IO技术包括 select poll

1.文件描述符集合:
select 使用三个文件描述符集合来表示待检查的文件描述符。这三个集合分别是读文件描述符集合(readfds)、写文件描述符集合(writefds)和异常文件描述符集合(exceptfds)。
2.超时设置:
select 允许设置一个超时时间,表示最长等待时间。当超时时间达到时,select 将返回,不再等待事件的发生。
3.轮询:
select 通过轮询检查文件描述符的状态,判断是否有可读、可写或异常事件发生。它会遍历指定的文件描述符集合,检查每个文件描述符的状态。
4.阻塞:
当没有任何文件描述符就绪时,select 可以阻塞程序,等待文件描述符变得可读、可写或发生异常。在这种情况下,它会一直等待,直到有文件描述符就绪或超时发生。
5.就绪文件描述符集合:
select 在返回时会修改传入的文件描述符集合,标识出哪些文件描述符已经就绪,可以进行相应的IO操作。

在使用 select 时,需要注意其效率问题,特别是在大规模的文件描述符集合中。因为 select 是线性扫描所有文件描述符的,当文件描述符数量增加时,性能可能会下降。在一些操作系统上,存在文件描述符数量的限制。
使用select实现同时读取键盘和鼠标数据,并且设置了3s超时 如果一直没有等到IO到达,就直接退出程序。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/select.h>

int main() {
    char buf[199];
    memset(buf, 0, sizeof(buf));

    // 打开鼠标设备
    int mouse_fd = open("/dev/input/mouse0", O_RDONLY);
    if (mouse_fd < 0) {
        perror("open mouse:");
        return -1;
    }
    struct timeval timeout;
    timeout.tv_sec = 3;   // 设置秒数
    timeout.tv_usec = 0;  // 设置微秒数
    printf("before select.\n");

    while (1) {
        fd_set read_fds;
        FD_ZERO(&read_fds);
        FD_SET(0, &read_fds);   // 标准输入(键盘)
        FD_SET(mouse_fd, &read_fds);  // 鼠标

        // 使用select监听多个文件描述符
        int result = select(mouse_fd + 1, &read_fds, NULL, NULL, &timeout);

        if (result > 0) {
            //判断出来是键盘IO到了
            if (FD_ISSET(0, &read_fds)) {
                // 从键盘读取
                read(0, buf, sizeof(buf));
                printf("keyboard 读出来的内容是:%s\n", buf);
                memset(buf, 0, sizeof(buf));
            }
            //判断出是鼠标到了
            if (FD_ISSET(mouse_fd, &read_fds)) {
                // 从鼠标读取
                read(mouse_fd, buf, 2);  // 从鼠标读取两位
                printf("mouse 读出来的内容是:%s\n", buf);
                memset(buf, 0, sizeof(buf));
            }
        }
        if(result == 0){
            printf("select 等待超时\n");
            return -1;
        }
    }

    // 关闭鼠标设备
    close(mouse_fd);

    return 0;
}

3.2.1多路复用IO poll原理

poll 是 Linux 中用于多路复用 I/O 操作的系统调用之一,它允许一个进程等待多个文件描述符上的事件发生。poll 的原理如下:
1.准备文件描述符集合和事件关注列表: 在调用 poll 之前,应用程序需要创建一个 struct pollfd 数组,该数组包含了要监听的文件描述符以及对每个文件描述符关注的事件。每个 struct pollfd 结构体包括以下字段:

  • fd
  • events
  • revents

2.调用 poll 函数: 应用程序调用 poll 函数,传递准备好的 struct pollfd 数组及数组的长度。

#include <poll.h>
int poll(struct pollfd *fds, nfds_t nfds, int timeout);
  • fds:指向struct pollfd数组的指针
  • nfds: 数组中结构体的数量
  • timeout:设置超时时间,-1代表一直等待

代码演示:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <poll.h>

int main() {
    char buf[199];
    memset(buf, 0, sizeof(buf));

    // 打开鼠标设备文件
    int mouse_fd = open("/dev/input/mouse0", O_RDONLY);
    if (mouse_fd < 0) {
        perror("open mouse:");
        return -1;
    }

    // 使用poll监听标准输入和鼠标输入
    struct pollfd fds[2];
    fds[0].fd = STDIN_FILENO;  // 标准输入
    fds[0].events = POLLIN ;  //事件为可读

    fds[1].fd = mouse_fd;  // 鼠标输入
    fds[1].events = POLLIN ;//事件为可读

    printf("Waiting for input...\n");

    while (1) {
        //这里的timeout 设置-1 代表一直等待
        int result = poll(fds, 2, -1);  // 阻塞等待事件发生

        if (result > 0) {
            if (fds[0].revents & (POLLIN | POLLPRI)) {
                // 从标准输入读取数据
                read(STDIN_FILENO, buf, sizeof(buf));
                printf("Keyboard input: %s\n", buf);
            }

            if (fds[1].revents & (POLLIN | POLLPRI)) {
                // 从鼠标设备读取数据
                read(mouse_fd, buf, sizeof(buf));
                printf("Mouse input: %s\n", buf);
            }
        } else if (result < 0) {
            perror("poll:");
            break;
        }
    }

    // 关闭文件描述符
    close(mouse_fd);

    return 0;
}

与select效果一致
在这里插入图片描述

3.3异步IO

在Linux中,异步IO(Asynchronous I/O)是一种文件IO操作的模型,与传统的同步IO模型(例如使用read和write函数)不同。在异步IO模型中,IO操作的请求被提交后,程序可以继续执行其他任务,而无需等待IO操作完成。
异步IO的关键特点包括:

  • 非阻塞: 异步IO允许程序在等待IO操作完成的同时继续执行其他任务,不会阻塞整个进程。

  • 回调机制: 异步IO通常通过回调机制来处理IO操作完成的通知。当IO操作完成时,系统会调用预先注册的回调函数,以便程序可以处理IO的结果。

  • 提高并发性: 异步IO适用于需要同时处理大量IO操作的场景,能够提高程序的并发性能。
    在Linux中,异步IO可以通过以下几种机制来实现:

  • fcntl+signal:使用fcntl中的F_SETOWN和O_ASYNC选项来设置异步IO的所有者,然后结合signal来注册信号处理函数,以便在IO事件发生时得到通知。

  • aio_ 函数族:* 提供了一组异步IO相关的系统调用,例如aio_read、aio_write等。这些函数使用结构体struct aiocb来描述IO操作,并可以设置回调函数。

  • epoll: epoll本身是一个多路复用机制,但也可以与异步IO结合使用,通过epoll监听IO事件,当IO操作完成时,通过回调机制处理。

  • libuv: 是一个跨平台的异步IO库,它封装了不同操作系统的异步IO机制,使得在不同平台上能够使用相似的异步IO接口。

使用fcntl+signa完成异步IO操作

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include <unistd.h>
#include <fcntl.h>
#include<signal.h>
int mouse_fd = -1;
//设置信号回调函数  鼠标时间设置为一个异步IO
void func(int sig){
    char buf[100] = {0};
    if (sig !=SIGIO)
    read (mouse_fd,buf,2);  //从鼠标读取两位
    printf("mouse读出来的内容是 :【%s】\n",buf);
}
int main(){
    char buf[100] = {0};
    //读取鼠标
    int flag = -1;
    mouse_fd = open("/dev/input/mouse0",O_RDONLY);
    if(mouse_fd<0){
        perror("open:");
        return -1;
    }
    //注册异步通知 把鼠标设置为异步IO事件
    flag  = fcntl(mouse_fd,F_GETFL);
    flag |= O_ASYNC;
    fcntl(mouse_fd,F_SETFL,flag);
    //把当前进程设置异步IO接收的进程
    fcntl(mouse_fd,F_SETOWN,getpid());
    // 注册信号处理函数
    signal(SIGIO, func);
    memset(buf,0,sizeof(buf));
    printf("before keyboard read.\n");
    read (0,buf,2);  //从键盘读取两个字节 就是0号文件标识符 read 本身就是阻塞式的
    printf("keyboard 读出来的内容是 :【%s】\n",buf);
    return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

国产麒麟系统开机没有网络需要点一下这个设置

问题描述&#xff1a; 一台国产电脑网线连接正常&#xff0c;打开网页后显示无法访问&#xff0c;那么是什么原因无法上网呢&#xff1f;下面就告诉你一个小方法去解决一下这个问题&#xff1b; 检查故障&#xff1a; 检测交换机、网线、水晶头全都正常&#xff0c;同房间摆放的…

ZooKeeper 实战(二) 命令行操作篇

文章目录 ZooKeeper 实战(二) 命令行操作篇1. 服务端命令1.1. 服务启动1.2. 查看服务1.3. 重启服务1.4. 停止服务 2. 客户端命令2.1. 启动客户端2.2. 查看节点信息查看根节点详情 ls -s /添加一个watch监视器 ls -w /列举出节点的级联节点 ls -R / 2.3. 查看节点状态2.4. 创建节…

Jenkins 问题

从gitlab 仓库拉去代码到Jenkins本地报错 ERROR: Couldn’t find any revision to build. Verify the repository and branch configuration for this job. 问题原因&#xff1a; 创建条目》配置的时候&#xff0c;gitlab仓库不存在master分支 修复后&#xff1a;

44-js return返回值,全局作用域,局部作用域,隐式作用域,变量的生命周期,delete释放内存

1.return返回值&#xff1a;函数执行后剩下结果就是返回值。 function fn(a,b,c){//return返回值return(abc);// console.log("aaa"); //return之后的值都不在执行了// alert("bbb"); //return之后的值不在执行了}console.log(fn(1,2,3)*10)…

人工智能:我的学习之旅与认知探索(第1版)

&#x1f31f;&#x1f30c; 欢迎来到知识与创意的殿堂 — 远见阁小民的世界&#xff01;&#x1f680; &#x1f31f;&#x1f9ed; 在这里&#xff0c;我们一起探索技术的奥秘&#xff0c;一起在知识的海洋中遨游。 &#x1f31f;&#x1f9ed; 在这里&#xff0c;每个错误都…

what is BERT?

BERT Introduction Paper 参考博客 9781838821593_ColorImages.pdf (packt-cdn.com) Bidirectional Encoder Representation from Transformer 来自Transformer的双向编码器表征 基于上下文&#xff08;context-based&#xff09;的嵌入模型。 那么基于上下文&#xff08;…

golang学习笔记——go语言多文件项目运行的四种方式

go语言多文件运行技巧 有两个源码文件的go语言项目如何运行? go.modmain.go Trie.go 如何直接运行go run main.go会提示找不到文件。 # 在windows10下运行 $ go run main.go # command-line-arguments .\main.go:6:9: undefined: Constructor是真的找不到文件吗。其实不是。…

一个成功的camera案例:ros2+gazebo+摄像头

各位看&#xff1a;随着大物体的移动&#xff0c;在涉嫌头的位置也发生了改变-----右上角那个/camera的位置也变了 右上角那个是摄像头图案&#xff0c;以下是仓库链接&#xff1a; ros-ign-gazebo-camera: https://github.com/arashsm79/ros-ign-gazebo-camera.git一个ros2摄…

基于多智能体点对点转换的分布式模型预测控制

matlab2020正常运行 基于多智能体点对点转换的分布式模型预测控制资源-CSDN文库

Zabbix“专家坐诊”第223期问答汇总

来源&#xff1a;乐维社区 问题一 Q&#xff1a;Zabbix 5.0安装完mysql之后怎么备份&#xff1f;忘记mysql当时创建的密码了&#xff0c;怎么样能查看设置的密码&#xff1f; A&#xff1a;mysql初始化密码在 /var/log/mysqld.log中可以看到&#xff0c;搜关键字temporary pas…

膜结构球形影院为观众打造观影新体验

在数字科技快速发展的当下&#xff0c;轻空间公司打破传统影院的束缚&#xff0c;领航未来娱乐体验的创新浪潮。膜结构球形影院问世&#xff0c;它不仅仅是一个娱乐场所&#xff0c;更是一场極致沉浸感的感官之旅&#xff0c;为观众带来震撼性的视听冲击。 沉浸式体验的新纪元 …

Jenkins安装和配置

拉取Jenkins镜像 docker pull jenkins/jenkins 编写jenkins_docker.yml version: "3.1" services:jenkins:image: jenkins/jenkinscontainer_name: jenkinsports:- 8080:8080- 50000:50000volumes:- ./data/:/var/jenkins_home/首次启动会因为数据卷data目录没有权限…

Nginx配置负载均衡实例

Nginx配置反向代理实例二 提醒一下&#xff1a;下面实例讲解是在Mac系统演示的&#xff1b; 负载均衡实例实现的效果 浏览器地址栏输入地址http://192.168.0.101/test/a.html&#xff0c;刷新页面进行多次请求&#xff0c;负载均衡效果&#xff0c;平均分配到8080端口服务和8…

DNS解析和它的三个实验

一、DNS介绍 DNS&#xff1a;domain name server 7层协议 名称解析协议 tcp /53 主从之间的同步 udp/53 名字解析 DNS作用&#xff1a;将域名转换成IP地址的协议 1.1DNS的两种实现方式 1.通过hosts文件&#xff08;优先级最高&#xff09; 分散的管理 linux /etc/hos…

资源三号03星-立体测绘卫星星座

资源三号03星作为我国民用高分辨率立体测图卫星资源三号系列的第三颗卫星&#xff0c;在资源三号02星技术状态的基础上进行了继承和适当优化&#xff0c;设计寿命由资源三号02星的5年延长至8年&#xff0c;星上搭载了三线阵立体测绘相机、多光谱相机和业务化应用的激光测高仪&a…

【JAVA】concurrentHashMap和HashTable有什么区别

&#x1f34e;个人博客&#xff1a;个人主页 &#x1f3c6;个人专栏&#xff1a;JAVA ⛳️ 功不唐捐&#xff0c;玉汝于成 目录 前言 正文 同步性质&#xff1a; 性能&#xff1a; 允许空键值&#xff08;Allow Nulls&#xff09;&#xff1a; 迭代器&#xff08;Iter…

066:vue中实现二维数组的全选、全不选、反选、部分全选功能(图文示例)

第061个 查看专栏目录: VUE ------ element UI 专栏目标 在vue和element UI联合技术栈的操控下,本专栏提供行之有效的源代码示例和信息点介绍,做到灵活运用。 (1)提供vue2的一些基本操作:安装、引用,模板使用,computed,watch,生命周期(beforeCreate,created,beforeM…

Ansible自动化运维

1.Ansible 介绍和架构 1.1 Ansible 发展历史 作者&#xff1a;Michael DeHaan&#xff08; Cobbler 与 Func 作者&#xff09; ansible 的名称来自科幻小说《安德的游戏》中跨 越时空的即时通信工具&#xff0c;使用它可以在相距数光年的距离&#xff0c;远程实时控制前线的舰…

【J-Flash基本使用总结】

【J-Flash基本使用总结】 VX&#xff1a;hao541022348 ■ 烧录文件■ 创建新的工程■ 烧录模式-SWD模式■ 烧录 ■ 读单片机Flash数据■ 设置或解除读保护 ■ 烧录文件 ■ 创建新的工程 接着点击 Options –> Project Settings…&#xff0c;选择CPU > Device &#xff…

强化学习应用(三):基于Q-learning的无人机物流路径规划研究(提供Python代码)

一、Q-learning简介 Q-learning是一种强化学习算法&#xff0c;用于解决基于马尔可夫决策过程&#xff08;MDP&#xff09;的问题。它通过学习一个价值函数来指导智能体在环境中做出决策&#xff0c;以最大化累积奖励。 Q-learning算法的核心思想是通过不断更新一个称为Q值的…