提问人:Heinrich supports Monica 提问时间:8/13/2013 最后编辑:Heinrich supports Monica 更新时间:8/28/2018 访问量:13254
plain C:使用 fopen() 打开一个目录
plain C: opening a directory with fopen()
问:
我有一个程序可以打开文件并检查其长度。
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()
-1
size_t
0xFFFFFFFFFFFFFFFF
不幸的是,上面代码()中的条件没有捕捉到这种情况,也没有(编辑:应该是)。-以 ord 作为格式字符串的命令显示比较的两边应具有相同的值。flen == ((size_t)-1)
flen == 0xFFFFFFFF
0xFFFFFFFFFFFFFFFF
printf()
%x
%d
为什么比较运算符的行为方式如此奇怪,即使两边是同一类型()?我使用 gcc 4.8.1 作为编译器。size_t
答:
目录在 C99 标准(或 C2011 标准)中不存在。因此,根据定义,-ing 目录是特定于实现或未定义的行为。fopen
fopen(3) 可能会失败 (给出结果)。fseek(3) 也可能失败 (通过返回 -1)。然后你最好检查 errno(3) 或使用 perror(3)NULL
ftell
记录为返回 ,并在失败时返回。在 64 位 Linux 上,这是 .long
-1L
0xffffffffffffffff
你的代码应该是
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_MAX
即0x7fffffffffffffff
顺便说一句,在 Linux 上,目录是文件的特例。在文件路径上使用 stat(2) (以及文件描述符, 可能使用 fileno(3) 从某些文件获取) 来了解有关某个文件的更多元数据, 包括它的 “类型” (通过其模式)。你希望 opendir(3)、readdir(3) 和 closedir(3) 对目录内容进行操作。 参见 inode(7)。fstat
FILE*
评论
int32_t
long
来自 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")
评论
sizeof(size_t)-1
FILE fd
FILE *fd
ftell()
size_t
long
0xFFFFFFFF
0xFFFFFFFFFFFFFFFF
open()
fopen()
fchdir()
opendir()
readdir()
fstat(fileno(fd))
fd
FILE *
FILE *fp = fopen("file", "r"); int fd = fileno(fp);