最近看到软文,总结了下同步与异步&阻塞与非阻塞的区别,由于习惯性的喜欢看文章后面的评论。发现有对原文作者的说法表示了质疑。让我这个小菜也不知道,作者的说法到底准不准确,于是乎开始在网上找找是否有“权威”通俗易懂的说法。不得不说,人民群众的力量是强大的,从不同的层级论述了——同步、异步、阻塞和非阻塞。现摘抄几个自己觉得理解到位的说法。
原1:http://my.oschina.net/haoran100/blog/707197
内容:
摘要
一直为同步异步,阻塞非阻塞概念所困扰,特定总结了下,原来是这么个意思
一直为同步异步,阻塞非阻塞概念所困扰,特定总结了下
一、同步与异步的区别
1、概念介绍
-
同步:所谓同步是一个服务的完成需要依赖其他服务时,只有等待被依赖的服务完成后,依赖的服务才能算完成,这是一种可靠的服务序列。要么成功都成功,失败都失败,服务的状态可以保持一致。
-
异步:所谓异步是一个服务的完成需要依赖其他服务时,只通知其他依赖服务开始执行,而不需要等待被依赖的服务完成,此时该服务就算完成了。至于被依赖的服务最终是否真正完成,无法确定,所以它是不可靠的服务序列。
2、消息通知
-
同步:当一个同步调用发出后,调用者要一直等待返回消息(或者调用结果)通知后,才能进行后续的执行;
-
异步:当一个异步过程调用发出后,调用者不能立刻得到返回消息(结果)。实际处理这个调用的部件在完成后,通过消息回调来通知调用者是否调用成功。
PS:调用者获取依赖服务异步回调结果一般有两种方式:一种是主动去轮训查询异步回调的结果,一种调用依赖服务时传入一个callback方法或者回调地址,依赖服务完成之后去调用callback通知调用者。一般情况,这两种方式都要支持才是一种良好的异步回调设计方法。
3、场景比喻
小明去买奶茶,可能会有两种方式
- 小明点单交钱,然后等着取奶茶;
- 小明点单交钱,然后奶茶妹给了小明一个小票,等小明的奶茶做好了,再告诉小明来取;
第一种方式就是同步,就等着奶茶妹做好奶茶,奶茶做好之后,小明拿到奶茶才算完成整个任务第二种方式就是异步,奶茶妹给了小明一个小票,小明就算完成了。至于最后奶茶做好没有,反正奶茶妹会告诉小明的,那是后面的事情了。
4、总结
同步与异步着重点在消息通知的方式,也就是调用结果通知的方式。结合场景就是,拿到奶茶的方式。
二、阻塞与非阻塞的区别
1、概念介绍
- 阻塞:阻塞调用是指调用结果返回之前,当前线程会被挂起,一直处于等待消息通知,不能够执行其他业务。函数只有在得到结果之后才会返回。
阻塞调用和同步调用不同的。对于同步调用来说,很多时候当前线程可能还是激活的,只是从逻辑上当前函数没有返回而已,此时,这个线程可能也会处理其他的消息。还有一点,在这里先扩展下:1.如果这个线程在等待当前函数返回时,仍在执行其他消息处理,那这种情况就叫做同步非阻塞;2.如果这个线程在等待当前函数返回时,没有执行其他消息处理,而是处于挂起等待状态,那这种情况就叫做同步阻塞;所以同步的实现方式会有两种:同步阻塞、同步非阻塞;同理,异步也会有两种实现:异步阻塞、异步非阻塞;对于阻塞调用来说,则当前线程就会被挂起等待当前函数返回;
- 非阻塞:非阻塞和阻塞的概念相对应,指在不能立刻得到结果之前,该函数不会阻塞当前线程,而会立刻返回。
虽然表面上看非阻塞的方式可以明显的提高CPU的利用率,但是也带了另外一种后果就是系统的线程切换增加。增加的CPU执行时间能不能补偿系统的切换成本需要好好评估。
2、场景比喻
还是小明去买奶茶,可能会有两种方式
- 2.1、小明点单交钱,然后等着取奶茶; ``` 1、小明在等着取奶茶的时候呢,啥都不干,就等着; 小明等奶茶的行为就叫做阻塞,小明在等奶茶的时候,阻塞了!而这种方式又叫做同步阻塞。
2、小明在等着取奶茶的时候呢,翻出手机,一会翻翻微信朋友圈,一会瞅瞅奶茶妹有没有做好奶茶; 小明这种行为就叫做非阻塞,因为没有阻塞在等奶茶这件事情上。而这种方式又叫做同步非阻塞。
- 2.2、小明点单交钱,然后奶茶妹给了小明一个小票,等小明的奶茶做好了,再告诉小明来取;
1、奶茶妹给了小明一个小票,然后小明还是傻等着; 小明这种拿到票还傻等着的行为,就叫阻塞。而这种行为,又叫做异步阻塞!这种最傻了。
2、奶茶妹给了小明一个小票,然后小明翻出手机一直刷朋友圈, 过了一会儿奶茶妹说小明你的奶茶好了,过来拿,小明才放下手机去拿奶茶。 这种方式就叫做非阻塞,又叫着异步非阻塞! ```
3、总结
阻塞与非阻塞的着重点在于当前线程等待消息返回的行为。换成场景就是,小明等奶茶的行为。
三、大总结
- 1、同步阻塞:小明啥都不干等奶茶。
- 2、同步非阻塞:小明一边玩手机一边等奶茶。
- 3、异步阻塞:小明拿着小票啥都不干等着奶茶妹告诉他奶茶好了
- 4、异步非阻塞:小明拿着小票玩着手机等着奶茶妹告诉他奶茶好了
原2:知乎 https://www.zhihu.com/question/19732473
说2.1:
作者:严肃
链接:https://www.zhihu.com/question/19732473/answer/20851256 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。阻塞”与"非阻塞"与"同步"与“异步"不能简单的从字面理解,提供一个从分布式系统角度的回答。
1.同步与异步 同步和异步关注的是消息通信机制 (synchronous communication/ asynchronous communication) 所谓同步,就是在发出一个*调用*时,在没有得到结果之前,该*调用*就不返回。但是一旦调用返回,就得到返回值了。 换句话说,就是由*调用者*主动等待这个*调用*的结果。 而异步则是相反,*调用*在发出之后,这个调用就直接返回了,所以没有返回结果。换句话说,当一个异步过程调用发出后,调用者不会立刻得到结果。而是在*调用*发出后,*被调用者*通过状态、通知来通知调用者,或通过回调函数处理这个调用。 典型的异步编程模型比如Node.js 举个通俗的例子: 你打电话问书店老板有没有《分布式系统》这本书,如果是同步通信机制,书店老板会说,你稍等,”我查一下",然后开始查啊查,等查好了(可能是5秒,也可能是一天)告诉你结果(返回结果)。 而异步通信机制,书店老板直接告诉你我查一下啊,查好了打电话给你,然后直接挂电话了(不返回结果)。然后查好了,他会主动打电话给你。在这里老板通过“回电”这种方式来回调。 2. 阻塞与非阻塞 阻塞和非阻塞关注的是程序在等待调用结果(消息,返回值)时的状态. 阻塞调用是指调用结果返回之前,当前线程会被挂起。调用线程只有在得到结果之后才会返回。 非阻塞调用指在不能立刻得到结果之前,该调用不会阻塞当前线程。 还是上面的例子, 你打电话问书店老板有没有《分布式系统》这本书,你如果是阻塞式调用,你会一直把自己“挂起”,直到得到这本书有没有的结果,如果是非阻塞式调用,你不管老板有没有告诉你,你自己先一边去玩了, 当然你也要偶尔过几分钟check一下老板有没有返回结果。 在这里阻塞与非阻塞与是否同步异步无关。跟老板通过什么方式回答你结果无关。 如果是关心blocking IO/ asynchronous IO, 参考 Unix Network Programming View Book说2.2:
作者:星辰大海
链接:https://www.zhihu.com/question/19732473/answer/26101328 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。一讲到网络编程的I/O模型,总会涉及到这几个概念。问了很多人,没几个能清晰地讲出他们之间的区别联系,甚至在网络上也有很多不同的观点,也不知是中国文字释义的博大精深,还是本来这几个概念就是绕人不倦。今天我也来给大家讲解一下我对这几个概念的理解。
既然网络上众说纷纭,不如找个权威参考一下,这个权威就是《UNIX网络编程:卷一》第六章——I/O复用。书中向我们提及了5种类UNIX下可用的I/O模型:
-
阻塞式I/O;
-
非阻塞式I/O;
-
I/O复用(select,poll,epoll...);
-
信号驱动式I/O(SIGIO);
-
异步I/O(POSIX的aio_系列函数);
阻塞式I/O模型:默认情况下,所有套接字都是阻塞的。怎么理解?先理解这么个流程,一个输入操作通常包括两个不同阶段:
(1)等待数据准备好;
(2)从内核向进程复制数据。 对于一个套接字上的输入操作,第一步通常涉及等待数据从网络中到达。当所有等待分组到达时,它被复制到内核中的某个缓冲区。第二步就是把数据从内核缓冲区复制到应用程序缓冲区。 好,下面我们以阻塞套接字的recvfrom的的调用图来说明阻塞标红的这部分过程就是阻塞,直到阻塞结束recvfrom才能返回。
非阻塞式I/O: 以下这句话很重要:进程把一个套接字设置成非阻塞是在通知内核,当所请求的I/O操作非得把本进程投入睡眠才能完成时,不要把进程投入睡眠,而是返回一个错误。看看非阻塞的套接字的recvfrom操作如何进行
可以看出recvfrom总是立即返回。
I/O多路复用:虽然I/O多路复用的函数也是阻塞的,但是其与以上两种还是有不同的,I/O多路复用是阻塞在select,epoll这样的系统调用之上,而没有阻塞在真正的I/O系统调用如recvfrom之上。如图
信号驱动式I/O:用的很少,就不做讲解了。直接上图
异步I/O:这类函数的工作机制是告知内核启动某个操作,并让内核在整个操作(包括将数据从内核拷贝到用户空间)完成后通知我们。如图:
注意红线标记处说明在调用时就可以立马返回,等函数操作完成会通知我们。
等等,大家一定要问了,同步这个概念你怎么没涉及啊?别急,您先看总结。 其实前四种I/O模型都是同步I/O操作,他们的区别在于第一阶段,而他们的第二阶段是一样的:在数据从内核复制到应用缓冲区期间(用户空间),进程阻塞于recvfrom调用。相反,异步I/O模型在这两个阶段都要处理。
再看POSIX对这两个术语的定义:
-
同步I/O操作:导致请求进程阻塞,直到I/O操作完成;
-
异步I/O操作:不导致请求进程阻塞。
好,下面我用我的语言来总结一下阻塞,非阻塞,同步,异步
-
阻塞,非阻塞:进程/线程要访问的数据是否就绪,进程/线程是否需要等待;
-
同步,异步:访问数据的方式,同步需要主动读写数据,在读写数据的过程中还是会阻塞;异步只需要I/O操作完成的通知,并不主动读写数据,由操作系统内核完成数据的读写。
说3:解释的例子不同
老张爱喝茶,废话不说,煮开水。
出场人物:老张,水壶两把(普通水壶,简称水壶;会响的水壶,简称响水壶)。 1 老张把水壶放到火上,立等水开。(同步阻塞) 老张觉得自己有点傻 2 老张把水壶放到火上,去客厅看电视,时不时去厨房看看水开没有。(同步非阻塞) 老张还是觉得自己有点傻,于是变高端了,买了把会响笛的那种水壶。水开之后,能大声发出嘀~~~~的噪音。 3 老张把响水壶放到火上,立等水开。(异步阻塞) 老张觉得这样傻等意义不大 4 老张把响水壶放到火上,去客厅看电视,水壶响之前不再去看它了,响了再去拿壶。(异步非阻塞) 老张觉得自己聪明了。 所谓同步异步,只是对于水壶而言。 普通水壶,同步;响水壶,异步。 虽然都能干活,但响水壶可以在自己完工之后,提示老张水开了。这是普通水壶所不能及的。 同步只能让调用者去轮询自己(情况2中),造成老张效率的低下。 所谓阻塞非阻塞,仅仅对于老张而言。 立等的老张,阻塞;看电视的老张,非阻塞。 情况1和情况3中老张就是阻塞的,媳妇喊他都不知道。虽然3中响水壶是异步的,可对于立等的老张没有太大的意义。所以一般异步是配合非阻塞使用的,这样才能发挥异步的效用。 ——来源网络,作者不明。 说4:作者:Shihui wang
链接:https://www.zhihu.com/question/19732473/answer/14413599 来源:知乎 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。其他楼层包括从技术角度都有了详细解释。这里主要是针对其他网友的疑问做的补充和修改,考虑到需要在编程概念上更严谨一点。
- 阻塞非阻塞表示下面 买书过程中 可能出现的状态,是从 我 这个单进程角度来看待这个买书这个问题。
- 同步异步表示一种协作方式,是从全局更高的角度 “进程之间 合作的方式” 来看待买书这个业务。两个进程之间如果商量采用异步方式处理买书这一业务,就不存在阻塞这种状态。
=============
A.概念阻塞非阻塞: 请求不能立即得到应答,需要等待,那就是阻塞;否则可以理解为非阻塞。
同步异步: 某业务需要甲乙甚至多方合作的时候,
- 总是按照“甲方请求一次,乙方应答一次”这样的有序序列处理业务,只有当“一次请求一次应答”的过程结束才可以发生下一次的“一次请求一次应答”,那么就说他们采用的是同步。(同步IO中,对同一个描述符的操作必须是有序的)
- 如果甲方只要有需要,就会发送请求,不管上次请求有没有得到乙方应答。而乙方只要甲方有请求就会接受,不是等这次请求处理完毕再接受甲方新请求。这样请求应答分开的序列,就可以认为是异步。异步情况下,请求和应答不需要一致进行,可能甲方后请求的业务,却先得到乙方的应答。同步是线性的,而异步可以认为是并发的。(异步IO中,异步IO可以允许多方同时对同一个描述符发送IO请求,或者一次发多个请求,当然有机制保证如何区分这些请求,)
举个例子:
- 我去买一本书,立即买到了,或者没有就走了,这就是非阻塞;(编程中设置IO成非阻塞,返回后再去检查描述符,或者等待通知,然后再去读取。相当于老板告诉我可以先忙点别的,过一会再来问问,或者老板通知我。但期间这个窗口(文件描述符)别人是用不了的)("立即买到了"在IO中也需要等待,不能算非阻塞IO)
- 如果恰好书店没有,我就等一直等到书店有了这本书买到了才走,这就是阻塞;而排在我后面的人呢只有我买到了书后才能再买书了。
- 如果书店恰好没有,我就告诉书店老板,书来了告诉我一声让我来取或者直接送到我家,然后我就走了,去做别的事了,这就是异步。这时候如果很多人来买书,都是老板登记一下完事。 (从IO角度来说,“告诉我来取”,这个近似于信号驱动IO,不能算异步IO。必须书送到我家才算是异步,如果不送到我家,我想看这本书之前,终究还是需要我跑一趟)
- 前面两种情况,非阻塞和阻塞都可以称为同步。
反映在编程方面就是 用户进程 调用 系统调用。(用户进程对应我,内核 对应 书店老板,书对应数据资源data , 买书就是一个系统调用了,其中内核拷贝数据到进程这个过程近似于老板送书到我手中)。
B. 在同步异步IO概念中,同步异步的不同在于,针对同一个描述符上的IO操作,从IO操作发起 到 得到 IO结果 这个过程而言,总是按照“发起请求,得到结果”这个有序序列进行的,这样便有了最小的等待这种情况:读取时,确知IO有数据,但需要等待内核拷贝数据到进程空间。这个最小情况的等待,同步IO都有。
unix网络编程中将IO模型分为5类:阻塞IO,非阻塞IO,IO复用,信号驱动,异步IO。
- 阻塞IO就是那种recv, read,一直等,等到有了数据才返回;
- 非阻塞IO就是立即返回,设置描述符为非阻塞,但是要进程自己一直检查是否可读;
- IO复用其实也是阻塞的,不过可以用来等很多描述符,比起阻塞有了进步,可以算有点异步了,但需要阻塞着检查是否可读。对同一个描述符的IO操作也是有序的。
- 信号驱动采用信号机制等待,有了更多的进步,不用监视描述符了,而且不用阻塞着等待数据到来,被动等待信号通知,由信号处理程序处理。但对同一个描述符的IO操作还是有序的。
- 异步IO,发送IO请求后,不用等了,也不再需要发送IO请求获取结果了。等到通知后,其实是系统帮你把数据读取好了的,你等到的通知也不再是要求你去读写IO了,而是告诉你IO请求过程已经结束了。你要做的就是可以处理数据了。且同一个描述符上可能同时存在很多请求。(对应上面那个买书例子中,就是送书到我家,我直接看书就行了,不需要再去跑一趟了)。
其中IO服用和信号驱动,在处理业务逻辑上可以说有异步,但在IO操作层面上来说还是同步的。
posix.1严格定义的异步IO是要求没有任何一点阻塞,而上述的前面四个(阻塞IO,非阻塞IO,IO复用,信号驱动)都不同程度阻塞了,而且都有一个共同的阻塞: 内核拷贝数据到进程空间的这段时间需要等待。 (所以上面的举例中: 必须要书送到我家,否则都不算异步,纠结。。。)
说5:
看看steven大叔的这篇文章。http://english.tebyan.net/newindex.aspx?pid=31159&BookID=23760&PageIndex=92&Language=3
同步和异步仅仅是关于所关注的消息如何通知的机制。同步的情况下,是由处理消息者自己去等待消息是否被触发,而异步的情况下是由触发机制来通知处理消息者 阻塞和非阻塞应该是发生在消息的处理的时刻。阻塞其实就是等待,发出通知,等待结果完成。非阻塞属于发出通知,立即返回结果,没有等待过程。 说6:不同场景,银行业务的例子来源:http://blog.chinaunix.net/uid-26000296-id-3754118.html
一、同步与异步
同步/异步, 它们是消息的通知机制1. 概念解释 A. 同步 所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回。 按照这个定义,其实绝大多数函数都是同步调用(例如sin isdigit等)。 但是一般而言,我们在说同步、异步的时候,特指那些需要其他部件协作或者需要一定时间完成的任务。 最常见的例子就是 SendMessage。 该函数发送一个消息给某个窗口,在对方处理完消息之前,这个函数不返回。 当对方处理完毕以后,该函数才把消息处理函数所返回的值返回给调用者。 B. 异步 异步的概念和同步相对。 当一个异步过程调用发出后,调用者不会立刻得到结果。 实际处理这个调用的部件是在调用发出后, 通过状态、通知来通知调用者,或通过回调函数处理这个调用。 以 Socket为例, 当一个客户端通过调用 Connect函数发出一个连接请求后,调用者线程不用等待结果,可立刻继续向下运行。 当连接真正建立起来以后,socket底层会发送一个消息通知该对象。 C. 三种返回结果途径 执行部件和调用者可以通过三种途径返回结果: a. 状态、 b. 通知、 c. 回调函数。 可以使用哪一种依赖于执行部件的实现,除非执行部件提供多种选择,否则不受调用者控制。 a. 如果执行部件用状态来通知, 那么调用者就需要每隔一定时间检查一次,效率就很低 有些初学多线程编程的人,总喜欢用一个循环去检查某个变量的值,这其实是一种很严重的错误。 b. 如果是使用通知的方式, 效率则很高,因为执行部件几乎不需要做额外的操作。 c. 至于回调函数, 和通知没太多区别。2. 举例说明 理解这两个概念,可以用去银行办理业务(可以取钱,也可以存钱)来比喻: 当到银行后, .可以去ATM机前排队等候 -- (排队等候)就是同步等待消息 .可以去大厅拿号,等到排到我的号时, 柜台的人会通知我轮到我去办理业务. -- (等待别人通知)就是异步等待消息. 在异步消息通知机制中, 等待消息者(在这个例子中就是等待办理业务的人)往往注册一个回调机制, 在所等待的事件被触发时由触发机制(在这里是柜台的人)通过某种机制(在这里是写在小纸条上的号码) 找到等待该事件的人. 在select/poll 等IO 多路复用机制中就是fd, 当消息被触发时,触发机制通过fd 找到处理该fd的处理函数.3. 在实际的程序中, 同步消息通知机制:就好比简单的read/write 操作,它们需要等待这两个操作成功才能返回; 同步, 是由处理消息者自己去等待消息是否被触发; 异步消息通知机制:类似于select/poll 之类的多路复用IO 操作, 当所关注的消息被触发时,由消息触发机制通知触发对消息的处理. 异步, 由触发机制来通知处理消息者; 还是回到上面的例子, 轮到你办理业务, 这个就是你关注的消息, 而办理什么业务, 就是对这个消息的处理, 两者是有区别的. 而在真实的IO 操作时: 所关注的消息就是 该fd是否可读写, 而对消息的处理是 对这个fd 进行读写. 同步/异步仅仅关注的是如何通知消息,它们对如何处理消息并不关心, 好比说,银行的人仅仅通知你轮到你办理业务了, 而办理业务什么业务(存钱还是取钱)他们是不知道的.二、阻塞与非阻塞 阻塞/非阻塞, 它们是程序在等待消息(无所谓同步或者异步)时的状态.1. 概念解释 A. 阻塞 阻塞调用是指调用结果返回之前,当前线程会被挂起。函数只有在得到结果之后才会返回。 有人也许会把阻塞调用和同步调用等同起来,实际上他是不同的。 对于同步调用来说,很多时候当前线程还是激活的,只是从逻辑上当前函数没有返回而已。 socket接收数据函数recv是一个阻塞调用的例子。 当socket工作在阻塞模式的时候, 如果没有数据的情况下调用该函数,则当前线程就会被挂起,直到有数据为止。 B. 非阻塞 非阻塞和阻塞的概念相对应,指在不能立刻得到结果之前,该函数不会阻塞当前线程,而会立刻返回。 C. 对象的阻塞模式和阻塞函数调用 对象是否处于阻塞模式和函数是不是阻塞调用有很强的相关性,但是并不是一一对应的。 阻塞对象上可以有非阻塞的调用方式,我们可以通过一定的API去轮询状态, 在适当的时候调用阻塞函数,就可以避免阻塞。 而对于非阻塞对象,调用特殊的函数也可以进入阻塞调用。函数select就是这样的一个例子。2. 举例说明 继续上面的那个例子, 不论是排队等待,还是使用号码等待通知, 如果在这个等待的过程中, . 等待者除了等待消息之外不能做其它的事情,那么该机制就是阻塞的, 表现在程序中,也就是该程序一直阻塞在该函数调用处不能继续往下执行. . 相反,有的人喜欢在银行办理这些业务的时候一边打打电话发发短信一边等待,这样的状态就是非阻塞的, 因为他(等待者)没有阻塞在这个消息通知上,而是一边做自己的事情一边等待.三、易混淆的点 很多人也会把异步和非阻塞混淆, 因为异步操作一般都不会在真正的IO 操作处被阻塞, 比如如果用select 函数,当select 返回可读时再去read 一般都不会被阻塞 就好比当你的号码排到时一般都是在你之前已经没有人了,所以你再去柜台办理业务就不会被阻塞. 可见,同步/异步与阻塞/非阻塞是两组不同的概念,它们可以共存组合, 而很多人之所以把同步和阻塞混淆,我想也是因为没有区分这两个概念, 比如阻塞的read/write 操作中,其实是把消息通知和处理消息结合在了一起, 在这里所关注的消息就是fd 是否可读/写,而处理消息则是对fd 读/写. 当我们将这个fd 设置为非阻塞的时候,read/write 操作就不会在等待消息通知这里阻塞, 如果fd 不可读/写则操作立即返回.四、同步/异步与阻塞/非阻塞的组合分析 _______阻塞____________________非阻塞_____ 同步 | 同步阻塞 同步非阻塞 异步 | 异步阻塞 异步非阻塞 同步阻塞形式: 效率是最低的, 拿上面的例子来说,就是你专心排队,什么别的事都不做。 实际程序中 就是未对fd 设置O_NONBLOCK 标志位的read/write 操作, 异步阻塞形式: 如果在银行等待办理业务的人采用的是异步的方式去等待消息被触发,也就是领了一张小纸条, 假如在这段时间里他不能离开银行做其它的事情,那么很显然,这个人被阻塞在了这个等待的操作上面; 异步操作是可以被阻塞住的,只不过它不是在处理消息时阻塞,而是在等待消息被触发时被阻塞. 比如select 函数, 假如传入的最后一个timeout 参数为NULL,那么如果所关注的事件没有一个被触发, 程序就会一直阻塞在这个select 调用处. 同步非阻塞形式: 实际上是效率低下的, 想象一下你一边打着电话一边还需要抬头看到底队伍排到你了没有, 如果把打电话和观察排队的位置看成是程序的两个操作的话, 这个程序需要在这两种不同的行为之间来回的切换,效率可想而知是低下的; 很多人会写阻塞的read/write 操作, 但是别忘了可以对fd 设置O_NONBLOCK 标志位,这样就可以将同步操作变成非阻塞的了; 异步非阻塞形式: 效率更高, 因为打电话是你(等待者)的事情,而通知你则是柜台(消息触发机制)的事情, 程序没有在两种不同的操作中来回切换. 比如说,这个人突然发觉自己烟瘾犯了,需要出去抽根烟, 于是他告诉大堂经理说,排到我这个号码的时候麻烦到外面通知我一下(注册一个回调函数), 那么他就没有被阻塞在这个等待的操作上面,自然这个就是异步+非阻塞的方式了. 如果使用异步非阻塞的情况, 比如aio_*组的操作,当发起一个aio_read 操作时,函数会马上返回不会被阻塞, 当所关注的事件被触发时会调用之前注册的回调函数进行处理,文章摘抄的有点多,但是集思广益,各路看客可以对号入座,综合分析。采用什么方式,只有适合自己的业务处理需求才行。