普通 C:使用 fopen() 打开目录
- 2024-11-11 08:27:00
- admin 原创
- 21
问题描述:
我有一个打开文件并检查其长度的程序。
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.
", argv[1]);
fclose(fd);
exit(1);
}
现在,至少在 Linux 下,fopen()
打开目录时会返回有效的文件描述符。这会导致查找操作返回-1
(或者,在 64 位系统上size_t
,无符号0xFFFFFFFFFFFFFFFF
=2 64 -1)。
不幸的是,上面代码中的条件(flen == ((size_t)-1)
)没有捕捉到这种情况,也没有flen == 0xFFFFFFFF
(编辑:应该是0xFFFFFFFFFFFFFFFF
)。-printf()
以%x
ord%d
作为格式字符串的命令表明比较的两边应该具有相同的值。
为什么比较运算符的行为如此奇怪,即使双方都是同一类型 ( size_t
)?我使用 gcc 4.8.1 作为编译器。
解决方案 1:
来自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 上,如果您尝试这样做,fopen("dirname", "wb")
您会得到 EISDIR 错误。我还尝试使用具有 d-------- 访问权限的目录,但仍然得到 EISDIR(而不是 EACCES)。
解决方案 2:
C99 标准(或 C2011 标准)中不存在目录。因此,根据定义,fopen
目录要么是特定于实现的,要么是未定义的行为。
fopen(3)可能会失败(给出NULL
结果)。fseek (3)也可能失败(返回 -1)。然后您最好检查errno(3)或使用perror(3)
ftell
据记载long
,如果-1L
失败则返回 和 。在 64 位 Linux 上,这是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); };
顺便说一句,在 Linux/Debian/Sid/AMD64 上使用 libc-2.17 和 3.10.6 内核时,该代码运行正常argv[1]
;/tmp
令人惊讶的flen
是,LONG_MAX
即0x7fffffffffffffff
顺便说一句,在 Linux 上,目录是文件的一个特例。在文件路径上使用stat(2)fstat
(以及文件描述符,可能使用fileno(3)从某些文件描述符获得)可以了解有关某些文件的更多元数据,包括其“类型”(通过其模式)。您希望opendir(3)、readdir(3)和closedir(3)对目录内容进行操作。另请参阅inode(7)。FILE*
- 2024年20款好用的项目管理软件推荐,项目管理提效的20个工具和技巧
- 2024年开源项目管理软件有哪些?推荐5款好用的项目管理工具
- 项目管理软件有哪些?推荐7款超好用的项目管理工具
- 项目管理软件哪个最好用?盘点推荐5款好用的项目管理工具
- 项目管理软件有哪些最好用?推荐6款好用的项目管理工具
- 项目管理软件有哪些,盘点推荐国内外超好用的7款项目管理工具
- 2024项目管理软件排行榜(10类常用的项目管理工具全推荐)
- 项目管理软件排行榜:2024年项目经理必备5款开源项目管理软件汇总
- 2024年常用的项目管理软件有哪些?推荐这10款国内外好用的项目管理工具
- 项目管理必备:盘点2024年13款好用的项目管理软件