C语言判断文件或文件夹是否存在

C语言判断文件或文件夹是否存在

汇总网上搜集的资料,总结了下述几种方法用于判断文件或者文件夹是否存在,注意每个函数的头文件引用。

一、使用access

  • 函数定义
#include <unistd.h>
int access(const char * pathname, int mode);
  • 函数说明

可用于判断文件或者目录是否存在, access()会检查是否可以读/写某一已存在的文件,access()只作权限的核查, 并不理会文件形态或文件内容,因此,如果一目录表示为"可写入",表示可以在该目录中建立新文件等操作,而非意味此目录可以被当做文件处理

pathname为文件或文件夹的路径,当前目录直接使用文件或文件夹名;mode表示检查模式,共4种模式:

  1. R_OK, 4 只判断是否有读权限,对应宏定义里面的00 只存在
  2. W_OK, 2 只判断是否有写权限,对应宏定义里面的02 写权限
  3. X_OK, 1 判断是否有执行权限,对应宏定义里面的04 读权限
  4. F_OK, 0 只判断是否存在,对应宏定义里面的05 读和写权限
  • 返回值

如果指定的存取方式有效,则函数返回0,否则函数返回-1

  • 使用示例
#include <unistd.h>
int file_exist(const char *path)
{
    return (access(path, F_OK) == 0);
}

二、使用stat

  • 函数定义
#include <sys/stat.h>
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以及fstatstatfstat完全相同,但fstat传入的是文件描述符;statlstat差别就在于l-linkstat会越过链接文件读取原文件,后者则读取链接文件

  • 返回值

执行成功则返回0,失败返回-1,错误代码存于errno(需要include <errno.h>

  • 使用示例
#include <sys/stat.h>
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 <errno.h>

  • 使用示例
#include <dirent.h>
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 核心内存不足。
*/

参考链接

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
©️2022 CSDN 皮肤主题:书香水墨 设计师:CSDN官方博客 返回首页
评论

打赏作者

SameWorld

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、C币套餐、付费专栏及课程。

余额充值