C语言判断文件或文件夹是否存在
汇总网上搜集的资料,总结了下述几种方法用于判断文件或者文件夹是否存在,注意每个函数的头文件引用。
一、使用access
-
函数定义
#include <>
int access(const char * pathname, int mode); -
函数说明
可用于判断文件或者目录是否存在, access()
会检查是否可以读/写某一已存在的文件,access()
只作权限的核查, 并不理会文件形态或文件内容,因此,如果一目录表示为"可写入",表示可以在该目录中建立新文件等操作,而非意味此目录可以被当做文件处理
pathname
为文件或文件夹的路径,当前目录直接使用文件或文件夹名;mode
表示检查模式,共4种模式:
-
R_OK, 4
只判断是否有读权限,对应宏定义里面的00 只存在
-
W_OK, 2
只判断是否有写权限,对应宏定义里面的02 写权限
-
X_OK, 1
判断是否有执行权限,对应宏定义里面的04 读权限
-
F_OK, 0
只判断是否存在,对应宏定义里面的05 读和写权限
- 返回值
如果指定的存取方式有效,则函数返回0
,否则函数返回-1
-
使用示例
#include <>
int file_exist(const char *path)
{
return (access(path, F_OK) == 0);
}
二、使用stat
-
函数定义
#include <sys/>
int stat(const char *file_name, struct stat *buf);
int fstat(int fd, struct stat *buf);
int lstat(const char *path, struct stat *buf); -
函数说明
可用于判断文件或者目录是否存在,通过文件名filename
获取文件信息,并保存在buffer
所指的结构体stat
中,根据stat
中的st_mode
,判断是不是目录文件。相关的还有lstat
以及fstat
;stat
与fstat
完全相同,但fstat
传入的是文件描述符;stat
与lstat
差别就在于l-link
,stat
会越过链接文件读取原文件,后者则读取链接文件
- 返回值
执行成功则返回0
,失败返回-1
,错误代码存于errno
(需要include <>
)
-
使用示例
#include <sys/>
int file_exist(const char *path)
{
struct stat buffer;
return (stat(path, &buffer)==0);
} -
扩展
/* stat结构体参数说明 /
struct stat
{
dev_t st_dev; //device 文件的设备编号
ino_t st_ino; //inode 文件的i-node
mode_t st_mode; //protection 文件的类型和存取的权限
nlink_t st_nlink; //number of hard links 连到该文件的硬连接数目, 刚建立的文件值为1.
uid_t st_uid; //user ID of owner 文件所有者的用户识别码
gid_t st_gid; //group ID of owner 文件所有者的组识别码
dev_t st_rdev; //device type 若此文件为装置设备文件, 则为其设备编号
off_t st_size; //total size, in bytes 文件大小, 以字节计算
unsigned long st_blksize; //blocksize for filesystem I/O 文件系统的I/O 缓冲区大小.
unsigned long st_blocks; //number of blocks allocated 占用文件区块的个数, 每一区块大小为512 个字节.
time_t st_atime; //time of lastaccess 文件最近一次被存取或被执行的时间, 一般只有在用mknod、utime、read、write 与tructate 时改变.
time_t st_mtime; //time of last modification 文件最后一次被修改的时间, 一般只有在用mknod、utime 和write 时才会改变
time_t st_ctime; //time of last change i-node 最近一次被更改的时间, 此参数会在文件所有者、组、权限被更改时更新
};
/
描述的st_mode 则定义了下列数种情况,可以使用“&”来读取文件的属性:
1、S_IFMT 0170000 文件类型的位遮罩
2、S_IFSOCK 0140000 scoket
3、S_IFLNK 0120000 符号连接
4、S_IFREG 0100000 一般文件
5、S_IFBLK 0060000 区块装置
6、S_IFDIR 0040000 目录
7、S_IFCHR 0020000 字符装置
8、S_IFIFO 0010000 先进先出
9、S_ISUID 04000 文件的 (set user-id on execution)位
10、S_ISGID 02000 文件的 (set group-id on execution)位
11、S_ISVTX 01000 文件的sticky 位
12、S_IRUSR (S_IREAD) 00400 文件所有者具可读取权限
13、S_IWUSR (S_IWRITE)00200 文件所有者具可写入权限
14、S_IXUSR (S_IEXEC) 00100 文件所有者具可执行权限
15、S_IRGRP 00040 用户组具可读取权限
16、S_IWGRP 00020 用户组具可写入权限
17、S_IXGRP 00010 用户组具可执行权限
18、S_IROTH 00004 其他用户具可读取权限
19、S_IWOTH 00002 其他用户具可写入权限
20、S_IXOTH 00001 其他用户具可执行权限上述的文件类型在 POSIX 中定义了检查这些类型的宏定义
21、S_ISLNK (st_mode) 判断是否为符号连接
22、S_ISREG (st_mode) 是否为一般文件
23、S_ISDIR (st_mode) 是否为目录
24、S_ISCHR (st_mode) 是否为字符装置文件
25、S_ISBLK (s3e) 是否为先进先出
26、S_ISSOCK (st_mode) 是否为socket 若一目录具有sticky 位 (S_ISVTX), 则表示在此目录下的文件只能被该文件所有者、此目录所有者或root 来删除或改名。错误代码:
1、ENOENT 参数file_name 指定的文件不存在
2、ENOTDIR 路径中的目录存在但却非真正的目录
3、ELOOP 欲打开的文件有过多符号连接问题, 上限为16 符号连接
4、EFAULT 参数buf 为无效指针, 指向无法存在的内存空间
5、EACCESS 存取文件时被拒绝
6、ENOMEM 核心内存不足
7、ENAMETOOLONG 参数file_name 的路径名称太长
*/
三、使用opendir
-
函数定义
#include <dirent.h>
DIR * opendir(const char * name); -
函数说明
用于判断目录是否存在,opendir()
用来打开参数name
指定的目录, 并返回DIR*
形态的目录流, 和open()
类似, 接下来对目录的读取和搜索都要使用此返回值。opendir
可用于判断目录是否存在,但是调用以后需要使用closedir()
关闭目录流,否则会有内存泄漏
- 返回值
成功则返回DIR*
型态的目录流, 打开失败则返回NULL
,错误代码存于errno
(需要include <>
)
-
使用示例
#include <>
int file_exist(const char *path)
{
if ((dp = opendir(path)) == NULL)
{
return 0;
}
else
{
closedir(dp);
return 1;
}
} -
扩展
/*
错误代码:
1、EACCESS 权限不足。
2、EMFILE 已达到进程可同时打开的文件数上限。
3、ENFILE 已达到系统可同时打开的文件数上限。
4、ENOTDIR 参数name 非真正的目录。
5、ENOENT 参数name 指定的目录不存在, 或是参数name 为一空字符串。
6、ENOMEM 核心内存不足。
*/
参考链接
- 判断目录是否存在并创建
- c语言 file如何判断文件是否存在
- C语言判断文件夹或者文件是否存在的方法
- 使用 C 语言中的 opendir 函数
- access (C语言函数名)
- C语言access()函数:判断是否具有存取文件的权限
- C语言stat()函数:获取文件状态
- C语言:stat、fstat、lstat函数使用方法
- C语言fstat()函数:由文件描述词取得文件状态
- C语言opendir()函数:打开目录函数