对于一次I/O访问,数据会先被拷贝到操作系统内核的缓冲区中,然后才会从操作系统内核的缓冲区拷贝到应用程序的地址空间。所以说,当一个read操作发生时,它会经历两个阶段:
1. 等待数据准备 (Waiting for the data to be ready)
2. 将数据从内核拷贝到进程中 (Copying the data from the kernel to the process)
正式因为这两个阶段,linux系统产生了下面五种网络模式的方案。
- 阻塞 I/O(blocking IO)
- 非阻塞 I/O(nonblocking IO)
- I/O 多路复用( IO multiplexing)
- 信号驱动 I/O( signal driven IO)
- 异步 I/O(asynchronous IO)
同步阻塞(blocking IO)在两个阶段内都是阻塞的,process此时只能等待无法做其他的事;(到银行取号并在窗口老老实实排队)
同步非阻塞在第一阶段是非阻塞的,process可以做别的事,但是在第二阶段数据从内核态到用户态时是阻塞的;(到银行取号在窗口排队,边排队边玩手机)
I/O 多路复用( IO multiplexing)又称事件驱动I/O,其好处是单个process就可以同时处理多个网络连接的I/O,当处理连接数较少时,这种方式和multi-threading + blocking IO的web server其实差不多;(取了多个窗口的号并老实排队,只要有一个窗口叫到了号就可以去办业务)
信号驱动I/O可以理解为,这个人插队的,不用排队就去办理业务。
异步I/O,用户进程发起read操作之后,立刻就可以开始去做其它的事。而另一方面,从kernel的角度,当它受到一个asynchronous read之后,首先它会立刻返回,所以不会对用户进程产生任何block。然后,kernel会等待数据准备完成,然后将数据拷贝到用户内存,当这一切都完成之后,kernel会给用户进程发送一个signal,告诉它read操作完成了。(银行VIP,全程无阻塞。既不需要取号排队,也不需要在窗口等待业务办理)