阻塞、非阻塞、同步和异步IO的区别是什么?

IO读取数据分为两个阶段,第一个阶段是内核准备好数据,第二个阶段是内核把数据从内核态拷贝到用户态。

阻塞IO是当用户调用 read 后,用户线程会被阻塞,等内核数据准备好并且数据从内核缓冲区拷贝到用户态缓存区后, read 才会返回。阻塞IO是两个阶段都会阻塞,没有数据时也会阻塞。

非阻塞IO是调用read后,如果没有数据就立马返回,通过不断轮询的方式去调用read,直到数据被拷贝到用户态的应用程序缓冲区,read请求才获取到结果。非阻塞IO阻塞的是第二个阶段,第一阶段没有数据时不会阻塞,第二阶段等待内核把数据从内核态拷贝到用户态的过程中才会阻塞。

同步 IO是应用程序发起一个 IO 操作后,必须等待内核把 IO 操作处理完成后才返回。无论 read 是阻塞 I/O,还是非阻塞 I/O, 都是同步调用,因为在 read 调用时,第二阶段内核将数据从内核空间拷贝到用户空间的过程都是需要等待的。

异步 IO应用程序发起一个 IO 操作后,调用者不能立刻得到结果,而是在内核完成 IO 操作后,通过信号或回调来通知调用者。异步 I/O 是内核数据准备好和数据从内核态拷贝到用户态这两个过程都不用等待。

总结一下,只有同步才有阻塞和非阻塞之分,异步必定是非阻塞的。

只有用户线程在操作IO的时候根本不去考虑IO的执行,全部都交给内核去完成, 而自己只等待一个完成信号的时候,才是真正的异步IO。select、poll、epool等IO多路复用方式都是同步的。