转载地址:https://blog.csdn.net/dutsoft/article/details/55224755

同步与异步
同步与异步的理解
同步与异步的重点在消息通知的方式上,也就是调用结果通知的方式。 
同步:当一个同步调用发出去后,调用者要一直等待调用结果的通知后,才能进行后续的执行。 
异步:当一个异步调用发出去后,调用者不能立即得到调用结果的返回。 
异步调用,要想获得结果,一般有两种方式: 
1、主动轮询异步调用的结果; 
2、被调用方通过callback来通知调用方调用结果。

生活实例
同步取快递:小明收到快递将送达的短信,在楼下一直等到快递送达。 
异步取快递:小明收到快递将送达的短信,快递到楼下后,小明再下楼去取。 
异步取快递,小明知道快递到达楼下有两种方式:1、不停的电话问快递小哥到了没有,即主动轮询;2、快递小哥到楼下后,打电话通知小明,然后小明下楼取快递,即回调通知。

阻塞和非阻塞
阻塞与非阻塞的理解
阻塞与非阻塞的重点在于进/线程等待消息时候的行为,也就是在等待消息的时候,当前进/线程是挂起状态,还是非挂起状态。 
- 阻塞阻塞调用在发出去后,在消息返回之前,当前进/线程会被挂起,直到有消息返回,当前进/线程才会被激活 
- 非阻塞非阻塞调用在发出去后,不会阻塞当前进/线程,而会立即返回。

生活实例
阻塞取快递:小明收到快递即将送达的信息后,什么事都不做,一直专门等快递。 
非阻塞取快递:小明收到快递即将送达的信息后,等快递的时候,还一边敲代码、一边刷微信。

同步与异步,重点在于消息通知的方式;阻塞与非阻塞,重点在于等消息时候的行为。 
所以,就有了下面4种组合方式

1. 同步阻塞:小明收到信息后,啥都不干,等快递;
2. 同步非阻塞:小明收到信息后,边刷微博,边等着取快递;
3. 异步阻塞:小明收到信息后,啥都不干,一直等着快递员通知他取快递;
4. 异步非阻塞:小明收到信息后,边刷着微博,边等快递员通知他取快递。

大部分程序的I/O模型都是同步阻塞的,单个进程每次只在一个文件描述符上执行I/O操作,每次I/O系统调用都会阻塞,直到完成数据传输。 
传统的服务器采用的就是同步阻塞的多进程模型。一个server采用一个进程负责一个request的方式,一个进程负责一个request,直到会话结束。进程数就是并发数,而操作系统支持的进程数是有限的,且进程数越多,调度的开销也越大,因此无法面对高并发。

Nginx采用了异步非阻塞的方式工作。那么Nginx是如何实现异步非阻塞的呢?那得先了解一下I/O多路复用。

I/O多路复用
所谓的I/O复用,就是多个I/O可以复用一个进程。I/O多路复用允许进程同时检查多个fd,以找出其中可执行I/O操作的fd。 
系统调用select()和poll()来执行I/O多路复用。在Linux2.6中引入的epoll()是select()的升级版,提供了更高的性能。通过I/O复用,我们可以在一个进程处理大量的并发I/O。

初级版I/O复用
比如一个进程接受了10000个连接,这个进程每次从头到尾的问一遍这10000个连接:“有I/O事件没?有的话就交给我处理,没有的话我一会再来问一遍。”然后进程就一直从头到尾问这10000个连接,如果这10000个连接都没有I/O事件,就会造成CPU的空转,并且效率也很低,不好不好。

那么,如果发明一个代理,每次能够知道哪个连接有了I/O流事件,不就可以避免无意义的空转了吗?为了避免CPU空转,可以引进了一个代理(一开始有一位叫做select的代理,后来又有一位叫做poll的代理,不过两者的本质是一样的)。

升级版I/O复用
select()
select可以同时观察许多流的I/O事件,在空闲的时候,会把当前线程阻塞掉,当有一个或多个流有I/O事件时,就从阻塞态中醒来,于是我们的程序就会轮询一遍所有的流(于是我们可以把“忙”字去掉了)。

while true {
  select(streams[])
  for i in streams[] {
            if i has data
                  read until unavailable
}
}

select()采用轮询的方式来检查fd是否就绪,当fd数量较多时,性能欠佳。因为从select那里仅仅知道了,有I/O事件发生了,但却并不知道是那几个流(可能有一个,多个,甚至全部),我们只能无差别轮询所有流,找出能读出数据,或者写入数据的流,对他们进行操作。–from 知乎

生活实例
小明家楼下有一个收发室,每次有快递到了就先代收,但收发室也不知道那个是小明的快递;但小明去取的时候,要查询所有代收的快递。

高级版I/O复用
epoll()
epoll能更高效的检查大量fd,UNIX中提供了类似功能的kqueue调用。 
epoll可以理解为event poll,不同于忙轮询和无差别轮询,当连接有I/O流事件产生的时候,epoll就会去告诉进程哪个连接有I/O流事件产生,然后进程就去处理这个事件。此时我们对这些流的操作都是有意义的。(复杂度降低到了O(k),k为产生I/O事件的流的个数,也有认为O(1)的)

生活实例
小明家楼下有一个收发室,每次有快递到了,就先代收并做了标记;然后通知小明去取送给小明的快递。

Nginx的异步非阻塞
Nginx配置use epoll后,以异步非阻塞方式工作,能够轻松处理百万级的并发连接。

events {
    worker_connections  1024;
    use kqueue;  # 在Linux中配置:use epoll;
}

在一个Web服务中,延迟最多的就是等待网络传输。nginx在启动后,会有一个master进程和多个worker进程。master进程主要用来管理worker进程,包含:接收来自外界的信号,向各worker进程发送信号,监控worker进程的运行状态,当worker进程退出后(异常情况下),会自动重新启动新的worker进程。而基本的网络事件,则是放在worker进程中来处理了。在一个请求需要等待的时候,worker可以空闲出来处理其他的请求,少数几个worker进程就能够处理大量的并发。

举例来说:同样的4个进程,如果采用一个进程负责一个request的方式;那么,同时进来4个request之后,每个进程就负责其中一个,直至会话关闭。期间,如果有第5个request进来了。就无法及时反应了,因为4个进程都没干完活呢,因此,一般有个调度进程,每当新进来了一个request,就新开个进程来处理。

nginx不这样,每进来一个request,会有一个worker进程去处理。但不是全程的处理,处理到什么程度呢?处理到可能发生阻塞的地方。比如向后端服务器转发request,并等待请求返回。那么,这个处理的worker不会这么傻等着,他会在发送完请求后,注册一个事件:“如果upstream返回了,告诉我一声,我再接着干”。于是他就休息去了。此时,如果再有request 进来,他就可以很快再按这种方式处理。而一旦上游服务器返回了,就会触发这个事件,worker才会来接手,这个request才会接着往下走。

总结
web server刚好属于网络io密集型应用,不算是计算密集型。web server的这种性质决定了每个request的大部份时间都消耗在网络传输中,实际上花费在server机器上的时间片不多。异步非阻塞,使用epoll,和大量细节处的优化,这就是Nginx几个进程就解决高并发的秘密所在。

Nginx的异步非阻塞相关推荐

  1. Nginx Note03——异步非阻塞机制

    曾经有一段时间,在我的理解中,同步就是阻塞,异步就是非阻塞.可不就是这样吗?从代码执行的角度来看,确实如此.但随着了解的深入,概念和场景的拓展,有了不一样的理解.试分析之,难免疏漏,欢迎指正. --题 ...

  2. nginx异步非阻塞理解

    nginx异步非阻塞理解? 看了一下午如何理解异步非阻塞,不是用生活举例的就是只是说某一点的,其实真的还是很不明白,最后只能通过nginx的异步非阻塞模型来稍微理解理解了,我也不拿什么生活的举例了因为 ...

  3. nginx异步非阻塞 解析

    同步与异步的理解 同步与异步的重点在消息通知的方式上,也就是调用结果通知的方式. 同步:当一个同步调用发出去后,调用者要一直等待调用结果的通知后,才能进行后续的执行. 异步:当一个异步调用发出去后,调 ...

  4. 处理大并发之一 对异步非阻塞的理解

    处理大并发之一 对异步非阻塞的理解 在研究nginx和node.js的时候常会遇到异步.非阻塞等,之前自己也经常使用epoll,对其同步与阻塞,异步与非阻塞有了一定的认识,现对参考资料总结下. 首先讨 ...

  5. 异步就是异步,根本就没有 异步非阻塞IO这个说法。阻塞 非阻塞,同步I/O 异步I/O 的区别

    先给大家安利一下这个 https://www.ibm.com/developerworks/cn/linux/l-async/ 里面关于 异步非阻塞IO 是错的. 异步就是异步,别扯 异步阻塞 异步非 ...

  6. 线程同步 阻塞 异步 非阻塞(转)

    同步:函数没有执行完不返回,线程被挂起 阻塞:没有收完数据函数不返回,线程也被挂起 异步:函数立即返回,通过事件或是信号通知调用者 非阻塞:函数立即返回,通过select通知调用者 这样看来异步和非阻 ...

  7. python3 异步 非阻塞 IO多路复用 select poll epoll 使用

    有许多封装好的异步非阻塞IO多路复用框架,底层在linux基于最新的epoll实现,为了更好的使用,了解其底层原理还是有必要的. 下面记录下分别基于Select/Poll/Epoll的echo ser ...

  8. 真正的 Tornado 异步非阻塞

    其中 Tornado 的定义是 Web 框架和异步网络库,其中他具备有异步非阻塞能力,能解决他两个框架请求阻塞的问题,在需要并发能力时候就应该使用 Tornado. 但是在实际使用过程中很容易把 To ...

  9. 同步阻塞,同步非阻塞,异步阻塞,异步非阻塞IO

    在高性能的I/O设计中,有两个比较著名的模式Reactor和Proactor模式,其中Reactor模式用于同步I/O,而Proactor运用于异步I/O操作. 在比较这两个模式之前,我们首先的搞明白 ...

最新文章

  1. 让vim正确显示cfg文件中的xxx = yyy=zzz
  2. 深入理解Spark Streaming执行模型
  3. JSP/Servlet中的汉字编码问题
  4. 地理素养的核心构成和主要特点
  5. 小心使用IOCP完成端口
  6. linux下qt对csv文件读取,qt 写入csv文件
  7. 力荐iReaper/as a Weekend Systems Admin...为何我从来没用过GHOST?
  8. 深入理解JavaScript系列(31):设计模式之代理模式
  9. java一千_涨姿势:为什么Java中“1000=1000”为false,而”100=100“为true?
  10. Ubuntu屏幕分辨率设置
  11. Win To Go MacBook Pro (16-inch, 2019) Catalina 10.15.7 bootcamp 6.1.7748
  12. DDR4、LPDDR4、LPDDR4x区别及DDR拓展
  13. java.lang.IllegalArgumentException: Can not set xx field xx to jav问题解决
  14. 硅芯片拆解:早期 555 定时器芯片内部一览
  15. mac Segmentation Fault
  16. 解决POI导出Excel单元格内容换行问题
  17. dp动态规划刷题总结
  18. Gabor滤波器进行纹理分割
  19. 什么是模态分析?什么是振型?
  20. 微信小程序 登录与退出登录

热门文章

  1. JUC锁-ReentrantReadWrite(五)
  2. 算法训练营12-动态规划
  3. Codeup-问题 B: 采药
  4. 组合和聚合的区别,通俗易懂。
  5. 【最佳解法】剑指 Offer 32 - III. 从上到下打印二叉树 III
  6. 简洁易懂:c:out标签详解
  7. 【传智播客】JavaWeb程序设计任务教程 第一章练习答案
  8. 24行代码AC_蓝桥杯2019省赛 试题 D: 数的分解
  9. 计算机网络实验(华为eNSP模拟器)——第十章 Eth-Trunk(链路聚合)
  10. 搭建MySQL+MHA服务易错位置