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

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

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

    典型的异步编程模型比如Node.js。

    2016.4.17更新:

    POSIX对这两个术语的定义:

    同步I/O操作:导致请求进程阻塞,直到I/O操作完成

    异步I/O操作:不导致请求进程阻塞

    2. 阻塞与非阻塞

    阻塞和非阻塞关注的是程序在等待调用结果(消息,返回值)时的状态。

    阻塞调用是指调用结果返回之前,当前线程会被挂起。调用线程只有在得到结果之后才会返回。非阻塞调用指在不能立刻得到结果之前,该调用不会阻塞当前线程。

    关于阻塞/非阻塞 & 同步/异步更加形象的比喻

    老张爱喝茶,废话不说,煮开水。 出场人物:老张,水壶两把(普通水壶,简称水壶;会响的水壶,简称响水壶)。

    1. 老张把水壶放到火上,立等水开。(同步阻塞) 老张觉得自己有点傻

    2. 老张把水壶放到火上,去客厅看电视,时不时去厨房看看水开没有。(同步非阻塞) 老张还是觉得自己有点傻,于是变高端了,买了把会响笛的那种水壶。水开之后,能大声发出嘀~~~~的噪音。

    3. 老张把响水壶放到火上,立等水开。(异步阻塞) 老张觉得这样傻等意义不大

    4. 老张把响水壶放到火上,去客厅看电视,水壶响之前不再去看它了,响了再去拿壶。(异步非阻塞) 老张觉得自己聪明了。

    所谓同步异步,只是对于水壶而言。普通水壶,同步;响水壶,异步。虽然都能干活,但响水壶可以在自己完工之后,提示老张水开了。这是普通水壶所不能及的。同步只能让调用者去轮询自己(情况2中),造成老张效率的低下。

    所谓阻塞非阻塞,仅仅对于老张而言。立等的老张,阻塞;看视的老张,非阻塞。情况1和情况3中老张就是阻塞的,媳妇喊他都不知道。虽然3中响水壶是异步的,可对于立等的老张没有太大的意义。所以一般异步是配合非阻塞使用的,这样才能发挥异步的效用。

    Linux下的五种IO模型

    阻塞IO(blocking IO) 非阻塞IO (nonblocking IO) IO复用(select 和poll) (IO multiplexing) 信号驱动IO (signal driven IO (SIGIO)) 异步IO (asynchronous IO (the POSIX aio_functions))

    前四种都是同步,只有最后一种才是异步IO。

    阻塞IO模型

    在这个模型中,应用程序(application)为了执行这个read操作,会调用相应的一个system call,将系统控制权交给kernel,然后就进行等待(这其实就是被阻塞了)。kernel开始执行这个system call,执行完毕后会向应用程序返回响应,应用程序得到响应后,就不再阻塞,并进行后面的工作。

    非阻塞IO

    在linux下,应用程序可以通过设置文件描述符的属性O_NONBLOCK,IO操作可以立即返回,但是并不保证IO操作成功。也就是说,当应用程序设置了O_NONBLOCK之后,执行write操作,调用相应的system call,这个system call会从内核中立即返回。但是在这个返回的时间点,数据可能还没有被真正的写入到指定的地方。也就是说,kernel只是很快的返回了这个 system call(只有立马返回,应用程序才不会被这个IO操作blocking),但是这个system call具体要执行的事情(写数据)可能并没有完成。而对于应用程序,虽然这个IO操作很快就返回了,但是它并不知道这个IO操作是否真的成功了,为了知道IO操作是否成功,一般有两种策略:一是需要应用程序主动地循环地去问kernel(这种方法就是同步非阻塞IO);二是采用IO通知机制,比如:IO多路复用(这种方法属于异步阻塞IO)或信号驱动IO(这种方法属于异步非阻塞IO)。

    IO多路复用(异步阻塞IO)

    和之前一样,应用程序要执行read操作,因此调用一个system call,这个system call被传递给了kernel。但在应用程序这边,它调用system call之后,并不等待kernel的返回结果而是立即返回,虽然立即返回的调用函数是一个异步的方式,但应用程序会被像select()、poll和epoll等具有复用多个文件描述符的函数阻塞住,一直等到这个system call有结果返回了,再通知应用程序。也就是说,“在这种模型中,IO函数是非阻塞的,使用阻塞 select、poll、epoll系统调用来确定一个 或多个IO 描述符何时能操作。”所以,从IO操作的实际效果来看,异步阻塞IO和第一种同步阻塞IO是一样的,应用程序都是一直等到IO操作成功之后(数据已经被写入或者读取),才开始进行下面的工作。不同点在于异步阻塞IO用一个select函数可以为多个描述符提供通知,提高了并发性。举个例子:假如有一万个并发的read请求,但是网络上仍然没有数据,此时这一万个read会同时各自阻塞,现在用select、poll、epoll这样的函数来专门负责阻塞同时监听这一万个请求的状态,一旦有数据到达了就负责通知,这样就将之前一万个的各自为战的等待与阻塞转为一个专门的函数来负责与管理。与此同时,异步阻塞IO和第二种同步非阻塞IO的区别在于:同步非阻塞IO是需要应用程序主动地循环去询问是否有操作数据可操作,而异步阻塞IO是通过像select和poll等这样的IO多路复用函数来同时检测多个事件句柄来告知应用程序是否可以有数据操作。

、唐城: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实现批量转换文件编码(批转换编码示例) php使用递归与迭代实现快速排序示例 python使用scrapy解析js示例 使用scrapy实现爬网站例子和实现网络爬虫(蜘蛛)的步骤 python中的对象拷贝示例 python引用传递 python回调函数的使用方法 python解析文件示例 下载安装setuptool和pip linux安装pip     Python的函数嵌套的使用方法 使用cx_freeze把python打包exe示例 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 不能用怎么办