当前位置 主页 > 服务器问题 > Linux/apache问题 >

    Linux 下的五种 IO 模型详细介绍(3)

    栏目:Linux/apache问题 时间:2019-10-28 13:48

    信号驱动IO (signal driven IO (SIGIO))

    应用程序提交read请求的system call,然后,kernel开始处理相应的IO操作,而同时,应用程序并不等kernel返回响应,就会开始执行其他的处理操作(应用程序没有被IO操作所阻塞)。当kernel执行完毕,返回read的响应,就会产生一个信号或执行一个基于线程的回调函数来完成这次 IO 处理过程。

    从理论上说,阻塞IO、IO复用和信号驱动的IO都是同步IO模型。因为在这三种模型中,IO的读写操作都是在IO事件发生之后由应用程序来完成。而POSIX规范所定义的异步IO模型则不同。对异步IO而言,用户可以直接对IO执行读写操作,这些操作告诉内核用户读写缓冲区的位置,以及IO操作完成后内核通知应用程序的方式。异步IO读写操作总是立即返回,而不论IO是否阻塞的,因为真主的读写操作已经由内核接管。也就是说,同步IO模型要求用户代码自行执行IO操作(将数据从内核缓冲区读入用户缓冲区,或将数据从用户缓冲区写入内核缓冲区),而异步IO机制则是由内核来执行IO操作(数据在内核缓冲区和用户缓冲区之间的移动是由内核在后台完成的)。你可以这样认为,同步IO向应用程序通知的是IO就绪事件,而异步IO向应用程序通知的是IO完成事件。linux环境下,aio.h头文件中定义的函数提供了对异步IO的支持。

    异步IO (asynchronous IO (the POSIX aio_functions))

    异步IO与上面的异步概念是一样的, 当一个异步过程调用发出后,调用者不能立刻得到结果,实际处理这个调用的函数在完成后,通过状态、通知和回调来通知调用者的输入输出操作。异步IO的工作机制是:告知内核启动某个操作,并让内核在整个操作完成后通知我们,这种模型与信号驱动的IO区别在于,信号驱动IO是由内核通知我们何时可以启动一个IO操作,这个IO操作由用户自定义的信号函数来实现,而异步IO模型是由内核告知我们IO操作何时完成。为了实现异步IO,专门定义了一套以aio开头的API,如:aio_read.

    小结:前四种模型–阻塞IO、非阻塞IO、多路复用IO和信号驱动IO都属于同步模式,因为其中真正的IO操作(函数)都将会阻塞进程,只有异步IO模型真正实现了IO操作的异步性。

    IO复用

    为了解释这个名词,首先来理解下复用这个概念,复用也就是共用的意思,这样理解还是有些抽象,为此,咱们来理解下复用在通信领域的使用,在通信领域中为了充分利用网络连接的物理介质,往往在同一条网络链路上采用时分复用或频分复用的技术使其在同一链路上传输多路信号,到这里我们就基本上理解了复用的含义,即公用某个“介质”来尽可能多的做同一类(性质)的事,那IO复用的“介质”是什么呢?为此我们首先来看看服务器编程的模型,客户端发来的请求服务端会产生一个进程来对其进行服务,每当来一个客户请求就产生一个进程来服务,然而进程不可能无限制的产生,因此为了解决大量客户端访问的问题,引入了IO复用技术,即:一个进程可以同时对多个客户请求进行服务。也就是说IO复用的“介质”是进程(准确的说复用的是select和poll,因为进程也是靠调用select和poll来实现的),复用一个进程(select和poll)来对多个IO进行服务,虽然客户端发来的IO是并发的但是IO所需的读写数据多数情况下是没有准备好的,因此就可以利用一个函数(select和poll)来监听IO所需的这些数据的状态,一旦IO有数据可以进行读写了,进程就来对这样的IO进行服务。

    理解完IO复用后,我们在来看下实现IO复用中的三个API(select、poll和epoll)的区别和联系,select,poll,epoll都是IO多路复用的机制,IO多路复用就是通过一种机制,可以监视多个描述符,一旦某个描述符就绪(一般是读就绪或者写就绪),能够通知应用程序进行相应的读写操作。但select,poll,epoll本质上都是同步IO,因为他们都需要在读写事件就绪后自己负责进行读写,也就是说这个读写过程是阻塞的,而异步IO则无需自己负责进行读写,异步IO的实现会负责把数据从内核拷贝到用户空间。三者的原型如下所示:

我有一卡车的娇无处可撒 - 抖音女生个性签名可爱又好听 有仇必报有爱必抱 - 抖音最火女生个性签名2019最新 慢慢变好才是给自己最好的礼物 - 抖音个性签名女简短励志2019 嘲讽也好掌声也好做好自己 - 微信女生个性签名奋斗励志 我不是个温柔的人可不妨碍我喜欢你呀 - 2019网红都在发的QQ个性 确认过眼神是可以尬聊的人 - 快手个性签名女生可爱幽默 女生专属抖音个性签名大全 - 抖音个性签名女生简短小清新 我这个该死的漂亮女孩 - qq扩列宣言女生可爱撩人签名 你比一大推零食还有诱惑力 - qq女生个性签名可爱好听 女生个性签名独一无二 - 女生一句话签名可爱霸气 、唐城:SVN没有update和commit 、唐城:Idea 使用SVN教程(有图) 、唐城:IDEA集成git和使用步骤(有图) 、唐城:mysql 内连接、左连接(左外连接)、右连接(右外连接) 、唐城:Spring Boot框架入门教程(快速学习版) 、唐城:构建微服务:Spring boot 入门篇 、唐城:拿来就能跑,1200多套微信小程序源码-史上最全的不同行 、唐城:老兵的经验之谈,成长路上这个道理越早知道越好 、唐城:(转发)Java开发、工作经验面试总结、[推荐阅读] 、唐城:远程debug调试,小伙伴们都很惊讶! 、唐城:清华学姐熬了一个月肝出这份32W字Linux知识手册,在 Git 、唐城:高薪程序员也躲不过35岁这一关..当能力与年龄脱节,我们 晴天的专栏:怎样规划你毕业以后的人生 流年的博客:C++语言篇 字符串及字符数组练习 blackball1998的博客:自动封装前端参数 blackball1998的博客:转发和重定向中携带变量 blackball1998的博客:设置Session和Cookie blackball1998的博客:扩展Spring MVC blackball1998的博客:添加拦截器 blackball1998的博客:文件上传下载 blackball1998的博客:全局异常处理 blackball1998的博客:请求异步处理 zhang_sheng_nian的博客:前端传后台的时间格式(如:2019-05-01 ModelMapper从Map转实体,数据类型转换出错failed to convert ja Samuel丶Zhou的博客:“2021-01-30T16:00:00.000Z“: expected f string佳佳的博客:“2021-01-28 18:34:28”时间转化成时间戳格 Sock的博客:用C语言写一个简单的扫雷小游戏 Anton的博客:【C语言初阶】扫雷小游戏(C语言简易版) 404name的博客:【C语言】从零开始的C语言小游戏之路(总) 使用python的chardet库获得文件编码并修改编码 忘记ftp密码使用python ftplib库暴力破解密码的方法示例 python使用rsa加密算法模块模拟新浪微博登录 python检测lvs real server状态 pyqt和pyside开发图形化界面 pyside写ui界面入门示例 python3使用urllib示例取googletranslate(谷歌翻译) python使用beautifulsoup从爱奇艺网抓取视频播放 python生成指定长度的随机数密码 python写的一个文本编辑器 python实现批量转换文件编码(批转换编码示例) 怎么判断一个网站seo的好坏? 很多企业为什么都忽视网站运营甚至说放弃? php make test 报错怎么办 Springboot项目中运用vue+ElementUI+echarts前后端交互实现动态圆环图(推荐) php怎么删除对象中的元素 mac怎么修改php环境变量 Unity连接MySQL并读取表格数据的实现代码 Java正则表达式实现经纬度的合法性操作 SpringCloud Alibaba使用Seata处理分布式事务的技巧 详解php如何实现一个简单的图片边缘检测 浅谈StringEntity 和 UrlEncodedFormEntity之间的区别 你究竟是否适合网络创业 七牛云存储——与创业精神一起前行 关于JDK+Tomcat+eclipse+MyEclipse的配置方法 一文搞懂C#实现读写文本文件中的数据 JDK8中的HashMap初始化和扩容机制详解 .net core中编辑json配置文件的方法 Java中HashMap的初始容量设置方式 把自己打造一个高效率的站长 php artisan 不能用怎么办