plain C:使用 fopen() 打开一个目录

plain C: opening a directory with fopen()

提问人:Heinrich supports Monica 提问时间:8/13/2013 最后编辑:Heinrich supports Monica 更新时间:8/28/2018 访问量:13254

问:

我有一个程序可以打开文件并检查其长度。

FILE* fd = fopen(argv[1], "rb");
fseek(fd, 0, SEEK_END);
size_t flen = ftell(fd);
if (flen == ((size_t)-1)) {
    printf("%s is a directory.\n", argv[1]);
    fclose(fd);
    exit(1);
}

现在,至少在 Linux 下,打开目录时会返回有效的文件描述符。这会导致返回 seek 操作(或者,在 64 位系统上为 =264-1,如无符号)。fopen()-1size_t0xFFFFFFFFFFFFFFFF

不幸的是,上面代码()中的条件没有捕捉到这种情况,也没有(编辑:应该是)。-以 ord 作为格式字符串的命令显示比较的两边应具有相同的值。flen == ((size_t)-1)flen == 0xFFFFFFFF0xFFFFFFFFFFFFFFFFprintf()%x%d

为什么比较运算符的行为方式如此奇怪,即使两边是同一类型()?我使用 gcc 4.8.1 作为编译器。size_t

C Linux 类型 IO

评论

0赞 Scotty Bauer 8/13/2013
你不是说吗?sizeof(size_t)-1
0赞 8/13/2013
@ScottyBauer 不,他没有。
7赞 8/13/2013
此外,应为 .并且没有返回,但是.-1 不在 64 位上(假设 2 的补码),而是 .FILE fdFILE *fdftell()size_tlong0xFFFFFFFF0xFFFFFFFFFFFFFFFF
1赞 8/13/2013
@ScottyBauer谷歌“C型铸造”。
2赞 Jonathan Leffler 8/13/2013
你可以有一个目录(因此它),但这就是你能用它做的全部事情。目的是允许您在文件描述符上使用。要读取目录,需要使用 、 等。为了找到它的大小,你可以使用 - 尽管使用名称而不是文件描述符让我觉得命名很糟糕(我会使用)。open()fopen()fchdir()opendir()readdir()fstat(fileno(fd))fdFILE *FILE *fp = fopen("file", "r"); int fd = fileno(fp);

答:

8赞 Basile Starynkevitch 8/13/2013 #1

目录在 C99 标准(或 C2011 标准)中不存在。因此,根据定义,-ing 目录是特定于实现或未定义的行为。fopen

fopen(3) 可能会失败 (给出结果)。fseek(3) 也可能失败 (通过返回 -1)。然后你最好检查 errno(3) 或使用 perror(3)NULL

ftell记录为返回 ,并在失败时返回。在 64 位 Linux 上,这是 .long-1L0xffffffffffffffff

你的代码应该是

FILE* fd = fopen(argv[1], "rb");
if (!fd) 
  { perror(argv[1]); exit(EXIT_FAILURE); };
if (fseek(fd, 0, SEEK_END)<0) 
  { perror("fseek"); exit(EXIT_FAILURE); };
long flen = ftell(fd);
if (flen == -1L)
  { perror("ftell"); exit(EXIT_FAILURE); };

顺便说一句,在带有 libc-2.17 和 3.10.6 内核的 Linux/Debian/Sid/AMD64 上,当 argv[1]/tmp 时,代码运行正常;令人惊讶的是,弗伦LONG_MAX0x7fffffffffffffff

顺便说一句,在 Linux 上,目录是文件的特例。在文件路径上使用 stat(2) (以及文件描述符, 可能使用 fileno(3) 从某些文件获取) 来了解有关某个文件的更多元数据, 包括它的 “类型” (通过其模式)。你希望 opendir(3)、readdir(3) 和 closedir(3) 对目录内容进行操作。 参见 inode(7)。fstatFILE*

评论

0赞 Heinrich supports Monica 8/13/2013
谢谢,我现在已经通过将 flen 类型更改为 来解决它,这在 32 位和 64 位系统上运行良好(在 64 位系统上使用不起作用)。int32_tlong
9赞 user3689693 5/30/2014 #2

来自 http://pubs.opengroup.org/onlinepubs/7908799/xsh/fopen.html

The fopen() function will fail if: 
[EISDIR] The named file is a directory and mode requires write access.

至少在 Linux 上,如果您尝试这样做,则会出现 EISDIR 错误。我还尝试使用具有 d-------- 访问权限的目录,但我仍然得到 EISDIR(而不是 EACCES)。fopen("dirname", "wb")