根据我在The Open Group网站上阅读
然而,通过对FIFO进行实验,似乎不可能同时拥有阻塞I/O描述符和非阻塞I/O描述符(因此是否设置了
这让我想到:是否可能将非阻塞I/O描述符和阻塞I/O描述符分配给同一文件,如果可能的话,是否取决于文件类型(普通文件、FIFO、块特殊文件、字符特殊文件、套接字等)?
fcntl
、open
、read
和write
的内容,我的印象是无论是否在文件描述符上设置了O_NONBLOCK
,也就是使用了非阻塞I/O,这应该是文件描述符的属性而不是底层文件的属性。作为文件描述符的属性意味着,例如,如果我复制一个文件描述符或打开另一个指向同一文件的描述符,则我可以在一个描述符上使用阻塞I/O,在另一个描述符上使用非阻塞I/O。然而,通过对FIFO进行实验,似乎不可能同时拥有阻塞I/O描述符和非阻塞I/O描述符(因此是否设置了
O_NONBLOCK
是底层文件[即FIFO]的属性)。#include <errno.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc, char **argv)
{
int fds[2];
if (pipe(fds) == -1) {
fprintf(stderr, "`pipe` failed.\n");
return EXIT_FAILURE;
}
int fd0_dup = dup(fds[0]);
if (fd0_dup <= STDERR_FILENO) {
fprintf(stderr, "Failed to duplicate the read end\n");
return EXIT_FAILURE;
}
if (fds[0] == fd0_dup) {
fprintf(stderr, "`fds[0]` should not equal `fd0_dup`.\n");
return EXIT_FAILURE;
}
if ((fcntl(fds[0], F_GETFL) & O_NONBLOCK)) {
fprintf(stderr, "`fds[0]` should not have `O_NONBLOCK` set.\n");
return EXIT_FAILURE;
}
if (fcntl(fd0_dup, F_SETFL, fcntl(fd0_dup, F_GETFL) | O_NONBLOCK) == -1) {
fprintf(stderr, "Failed to set `O_NONBLOCK` on `fd0_dup`\n");
return EXIT_FAILURE;
}
if ((fcntl(fds[0], F_GETFL) & O_NONBLOCK)) {
fprintf(stderr, "`fds[0]` should still have `O_NONBLOCK` unset.\n");
return EXIT_FAILURE; // RETURNS HERE
}
char buf[1];
if (read(fd0_dup, buf, 1) != -1) {
fprintf(stderr, "Expected `read` on `fd0_dup` to fail immediately\n");
return EXIT_FAILURE;
}
else if (errno != EAGAIN) {
fprintf(stderr, "Expected `errno` to be `EAGAIN`\n");
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}
这让我想到:是否可能将非阻塞I/O描述符和阻塞I/O描述符分配给同一文件,如果可能的话,是否取决于文件类型(普通文件、FIFO、块特殊文件、字符特殊文件、套接字等)?