一、传统IO:read() + write()
发生4次用户态和内核态的上下文切换,4次拷贝。
1、用户进程通过read()方法向操作系统发起调用,此时上下文从用户态转向内核态
2、DMA控制器把数据从硬盘中拷贝到内核读缓冲区
3、CPU把内核读缓冲区数据拷贝到用户缓冲区,上下文从内核态转为用户态,read()返回
4、用户进程通过write()方法发起调用,上下文从用户态转为内核态
5、CPU将用户缓冲区中数据拷贝到socket缓冲区/内核写缓冲区
6、DMA控制器把数据从socket缓冲区/内核写缓冲区拷贝到网卡,上下文从内核态切换回用户态,write()返回
二、零拷贝
零拷贝技术是指计算机执行操作时,CPU不需要先将数据从某处内存复制到另一个特定区域,这种技术通常用于通过网络传输文件时节省CPU周期和内存带宽。
(1)mmap+write:使用mmap代替read,减少一次CPU拷贝,节省一半的内存空间。
(4次用户态和内核态的上下文切换和3次拷贝)
mmap主要实现方式是将内核读缓冲区的地址和用户缓冲区的地址进行映射,内核缓冲区和应用缓冲区共享,从而减少了从读缓冲区到用户缓冲区的一次CPU拷贝。
1、用户进程通过mmap()方法向操作系统发起调用,上下文从用户态转向内核态
2、DMA控制器把数据从硬盘中拷贝到内核读缓冲区
3、上下文从内核态转为用户态,mmap调用返回
4、用户进程通过write()方法发起调用,上下文从用户态转为内核态
5、CPU将内核读缓冲区中数据拷贝到socket缓冲区
6、DMA控制器把数据从socket缓冲区/内核写缓冲区拷贝到网卡,上下文从内核态切换回用户态,write()返回
(2)sendfile:使用sendfile代替read+write,减少了一次CPU拷贝,而且还减少了2次上下文切换。
(2次用户态和内核态的上下文切换和3次拷贝)
sendfile数据可以直接在内核空间进行传输,因此避免了用户空间和内核空间的拷贝,同时由于使用sendfile替代了read+write从而节省了一次系统调用,也就是2次上下文切换。
1、用户进程通过sendfile()方法向操作系统发起调用,上下文从用户态转向内核态
2、DMA控制器把数据从硬盘中拷贝到内核读缓冲区
3、CPU将内核读缓冲区中数据拷贝到socket缓冲区/内核写缓冲区
4、DMA控制器把数据从socket缓冲区/内核写缓冲区拷贝到网卡,上下文从内核态切换回用户态,sendfile调用返回