动力节点首页 全国咨询热线:400-8080-105

绑定手机号,登录
手机号

验证码

微信登录
手机号登录
手机号

验证码

微信登录与注册
微信扫码登录与注册

扫码关注微信公众号完成登录与注册
手机号登录
首页 > 文章

详解零拷贝技术

04-29 11:25 682浏览
举报 T字号
  • 大字
  • 中字
  • 小字

零拷贝技术是编写高性能服务器的一个关键技术,在正式讲零拷贝技术之前,我们先来说明一下两个概念: 用户空间与内核空间。

用户空间

通俗的说,用户空间 就是运行着用户编写的应用程序的虚拟内存空间。在32位的操作系统中,每个进程都有 4GB 独立的虚拟内存空间,而 0 ~ 3GB 的虚拟内存空间就是用户空间 。

内核空间

内核空间 就是运行着操作系统代码的虚拟内存空间,而 3GB ~ 4GB 的虚拟内存空间就是内核空间。

下图展示了用户空间与内核空间在进程虚拟内存空间所在的位置:

图片

为什么要介绍 用户空间 和 内核空间 呢?

我们先来回忆一下,服务端发送一个文件给客户端一般需要进行什么操作。一般来说,服务端发送一个文件给客户端的步骤如下:

  • 首先需要调用 read 读取文件的数据到用户空间缓冲区中。
  • 然后再调用 write 把缓冲区的数据发送给客户端 Socket。

伪代码如下:

while ((n = read(file, buf, 4069)) > 0) {
    write(sock, buf , n);
}

在上面的过程中,调用了 read 和 write 两个系统调用。read 系统调用是从文件中读取数据到用户空间的缓冲区中,所以调用 read 时需要从内核空间复制数据到用户空间,如图所示:

图片

上图就是数据的复制过程,首先会从文件中读取数据到内核的 页缓存(page cache),然后再从页缓存中复制到用户空间的缓冲区中。

而当调用 write 系统调用把用户空间缓冲区中的数据发送到客户端 Socket 时,首先会把缓冲区的数据复制到内核的 Socket 缓冲区中,网卡驱动会把 Socket 缓冲区的数据发送出去,如图 所示:

图片

从上图可以看出,服务端发送文件给客户端的过程中需要进行两次数据复制,第一次是从内核空间的页缓存复制到用户空间的缓冲区,第二次是从用户空间的缓冲区复制到内核空间的 Socket 缓冲区。

仔细观察我们可以发现,上图中的页缓存其实可以直接复制到 Socket 缓冲区,而不需要复制到用户空间缓冲区的。如图所示:

图片

如上图所示,不需要用户空间作为数据中转的技术叫 零拷贝技术。那么,我们可以通过哪个系统调用来实现上图中的技术呢?答案就是 sendfile,我们来看看 sendfile 系统调用的原型:

#include <sys/sendfile.h>

ssize\_t sendfile(int out\_fd, int in\_fd, off\_t \*offset, size\_t count);
复制代码

下面介绍一下 sendfile 各个参数的作用:

  • out_fd:数据接收方文件句柄(一般为 Socket 句柄)。
  • in_fd:数据提供方文件句柄(一般为文件句柄)。
  • offset:如果 offset 不为 NULL,表示从哪里开始发送数据的偏移量。
  • count:表示需要发送多少字节的数据。
  • sendfile 发送数据的过程如图 所示:

图片

对比一下,我们发现使用 sendfile 可以减少一次系统调用,并且减少一次数据拷贝过程。

本文主要通过 sendfile 系统调用来介绍零拷贝技术,但显而易见,零拷贝技术不单只有 sendfile,如 mmap、splice 和 直接I/O 等都是零拷贝技术的实现,有兴趣的可以参考动力节点在线的官方资料来学习。

0人推荐
共同学习,写下你的评论
0条评论
代码小兵286
程序员代码小兵286

45篇文章贡献179882字

相关课程 更多>

作者相关文章更多>

推荐相关文章更多>

JavaWeb的3大组件

代码小兵49806-11 15:28

全面解析Cookie技术

代码小兵49806-11 15:51

浅谈JavaWeb架构演变

代码小兵49806-11 16:22

探讨Web开发中的Session存储与管理

代码小兵51603-29 17:28

JavaScript基础知识

 暴风城-小飞04-06 20:49

发评论

举报

0/150

取消