在 linux 平台,如果想从文件描述符中读取数据,主要通过以下 API 实现,当然还有其他 API。
ssize_t read(int fd, void *buf, size_t nbyte);
ssize_t pread(int fd, void *buf, size_t nbyte, off_t offset);
ssize_t readv(int fd, const struct iovec *iov, int iovcnt);
ssize_t preadv(int fd, const struct iovec *iov, int iovcnt, off_t offset);
read 会从 fd 当前的 offset 处开始读,读取完 nbyte 后,该 fd 的 offset 会增加 nbyte(假设可以读取到 nbyte),下一次 read 则从新的 offset 处开始读。
而 pread 则是从指定的 offset 处开始读,这个 offset 是相对于 0 的一个绝对值,与 fd 当前的 offset 没有关系。
打开一个文件时,offset 默认为 0;如果打开时指定了 O_APPEND 选项,那么 offset 为 SEEK_END 即文件末尾。
read 和 pread 的区别就是,read 会改变 fd 的 offset,而 pread 不会改变。
pread 的实际操作类似于 lseek + read,即先将 offset 调整到指定值,再调用 read 读取数据,两者的区别在于 pread 是一个原子操作,从而可以保证一定是从指定的 offset 处开始读;另外就是 pread 读取完数据会恢复执行之前的 offset,即 pread 操作前后的 offset 是一致的。
readv 则是从 fd 中读取数据到多个 buf,buf 数为 iovcnt,每个 buf 有自己的长度(可以一样),一个 buf 写满(写指读出数据并保存),才接着写下一个 buf,依次类推。preadv 与 readv 的关系,与上述 read 和 pread 的关系一样。
除了文件 I/O,还有网络 I/O,read/write 也可用于网络 I/O,只用于网络 I/O 的 API 有:
ssize_t send(int sockfd, const void *buf, size_t len, int flags);
ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,
const struct sockaddr *dest_addr, socklen_t addrlen);
ssize_t sendmsg(int sockfd, const struct msghdr *msg, int flags);
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,
struct sockaddr *src_addr, socklen_t *addrlen);
ssize_t recvmsg(int sockfd, struct msghdr *msg, int flags);
read、write 和 recv、send 在网络 I/O 上并无本质区别,只是在参数上面有细微差别。
接下来的问题是,各个 API 的使用场景,后续完善。