1 阻塞IO模型html
从字面来理解,就是调用时可能被阻塞, 什么叫阻塞,要知道一个进程有N种状态,学过OS都知道 若是阻塞,就会把当前进程放在某个条件的阻塞队列里。 直到条件知足了,才会转移此进程进入就绪队列。 固然,就绪队列还有个优先级的概念,就不扯远了。 阻塞IO. 1)调用API,好比 recvfrom,从用户态进入内核态。 2) 内核发现无数据,将进程信息节点插入阻塞队列,等待数据。 3)进程一直阻塞,阻塞,阻塞。 4)数据到达,内核拷贝数据, 数据的流动路径是:网卡---socket内核缓冲区---用户的缓冲区。 5)拷贝完成后返回,咱们就能够获得recvfrom的结果,成功/失败。
2 非阻塞IO模型java
什么叫非阻塞,就是从用户角度来讲,函数当即能够获得结果:成功/失败。 1)用户调用recvfrom,内核发现无数据,当即返回结果。 2)用户调用recvfrom,内核发现无数据,当即返回结果。 3)用户调用recvfrom,内核发现无数据,当即返回结果。 4)用户调用recvfrom,内核发现有数据,将数据从socket内核缓冲区复制到 用户缓冲区,返回结果。
3 IO复用模型redis
1)将多个须要监听的fd给epoll,让其监控。 注意:epoll自己是阻塞的,能够设定时间。 2)epoll返回结果,用户能够知道哪些fd可读,可写,有异常等。 3)剩下的就是用户本身按需操做fd. 好比redis就是使用IO复用模型。
备注:epoll的最大fd个数:cat /proc/sys/fs/file-max,跟内存关系比较大
4)信号驱动模型网络
1)开启socket信号驱动IO功能, 经过sigaction设置信号处理函数。 2)当socket有数据时,发生一个SIGIO信号, 执行以前的信号处理函数。 3)用户逻辑放在信号处理函数里实现。
5)异步IO异步
1)告知内核启动某个操做,在内核完成全部的操做(包括复制数据到用户缓冲区)后
经过信号机制通知用户。 2)信号处理程序直接处理用户缓冲区里的数据,由于数据已经被内核复制完毕。 与信号驱动的区别是: 信号驱动经过信号告知,能够启动某个操做了。 异步IO通知用户:我连复制数据的步骤都替你完成了,用户只须要处理便可。
总结一下:socket
为何会出现这么多的模型?函数
需求来源于需求,咱们想一想在开发网络程序时,对一个socket须要关注哪些方面?spa
1)这个socket可读,可写,有异常?code
2)有数据可操控时,复制数据到用户缓冲区。htm
下面是我对这几种IO模型的理解
是否解决socket 可用的问题 |
是否自动将数据从套接字缓冲区 复制到用户缓冲区 |
评分 |
|
阻塞模型 |
× |
× |
★ |
非阻塞模型 |
× |
× |
★★ |
IO复用模型 |
√ |
× |
★★★ |
信号驱动 |
√ |
× |
★★★ |
异步IO |
√ |
√ |
★★★★★ |