C语言实现Linux下删除非空目录

 

#include <sys/stat.h>
#include <dirent.h>
#include <fcntl.h>

/**
* 递归删除目录(删除该目录以及该目录包含的文件和目录)
* @dir:要删除的目录绝对路径
*/
int remove_dir(const char *dir)
{
    char cur_dir[] = ".";
    char up_dir[] = "..";
    char dir_name[128];
    DIR *dirp;
    struct dirent *dp;
    struct stat dir_stat;

    // 参数传递进来的目录不存在,直接返回
    if ( 0 != access(dir, F_OK) ) {
        return 0;
    }

    // 获取目录属性失败,返回错误
    if ( 0 > stat(dir, &dir_stat) ) {
        perror("get directory stat error");
        return -1;
    }

    if ( S_ISREG(dir_stat.st_mode) ) {  // 普通文件直接删除
        remove(dir);
    } else if ( S_ISDIR(dir_stat.st_mode) ) {   // 目录文件,递归删除目录中内容
        dirp = opendir(dir);
        while ( (dp=readdir(dirp)) != NULL ) {
            // 忽略 . 和 ..
            if ( (0 == strcmp(cur_dir, dp->d_name)) || (0 == strcmp(up_dir, dp->d_name)) ) {
                continue;
            }

            sprintf(dir_name, "%s/%s", dir, dp->d_name);
            remove_dir(dir_name);   // 递归调用
        }
        closedir(dirp);

        rmdir(dir);     // 删除空目录
    } else {
        perror("unknow file type!");    
    }

    return 0;
}


int main()
{
    string dstpath = "/DATA/123";

    //如果目录存在就删除目录
    if (access(dstpath.c_str(), 0) == 0)  
    {
        cout << "remove " << dstpath << endl;
        int flag = remove_dir(dstpath.c_str()); 
        cout << flag << endl; 
    }
}

 

<think>我们正在讨论C语言删除空目录的方法。根据引用[2],我们知道C标准库中的rmdir函数只能删除空目录。要删除空目录,我们需要递归地删除目录中的所有内容(文件和子目录),然后再删除目录本身。 在C语言中,我们可以使用nftw(遍历文件树)函数来递归遍历目录树,并在遍历过程中删除每个文件和目录。以下是一个使用nftw函数删除空目录的示例: 步骤: 1. 包含必要的头文件:<ftw.h>、<stdio.h>、<unistd.h>等。 2. 定义一个回调函数,该函数将在遍历过程中被调用,用于删除每个文件或目录。 3. 在回调函数中,根据文件类型(文件或目录)进行删除操作。注意:在删除目录之前,需要确保该目录(即先删除其所有内容)。 4. 使用nftw函数遍历目录,并设置合适的标志(如FTW_DEPTH表示深度优先遍历,这样在删除目录时,其内容已被删除)。 注意:nftw函数是POSIX标准的一部分,因此在Linux环境下通常可用。 示例代码: ```c #define _XOPEN_SOURCE 700 // 启用POSIX.1-2008标准(包含nftw) #include <stdio.h> #include <stdlib.h> #include <ftw.h> #include <unistd.h> // 回调函数,用于删除每个文件/目录 static int remove_callback(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf) { int ret = remove(fpath); // 删除文件或空目录 if (ret) { perror(fpath); } return ret; } // 删除空目录的函数 int remove_directory(const char *path) { // 使用nftw遍历目录树,深度优先(FTW_DEPTH)确保先删除目录中的内容再删除目录 // 第四个参数是同时打开的文件描述符数量(这里设为64),最后一个参数是标志(FTW_DEPTH|FTW_PHYS) return nftw(path, remove_callback, 64, FTW_DEPTH | FTW_PHYS); } int main(int argc, char *argv[]) { if (argc != 2) { fprintf(stderr, "Usage: %s <directory>\n", argv[0]); exit(EXIT_FAILURE); } if (remove_directory(argv[1])) { perror("Error"); exit(EXIT_FAILURE); } return EXIT_SUCCESS; } ``` 说明: 1. `remove_callback`函数中,我们使用`remove`函数删除文件或空目录。`remove`函数在遇到文件时调用`unlink`,在遇到目录时调用`rmdir`。由于我们使用深度优先遍历(FTW_DEPTH),所以当回调函数处理一个目录时,该目录下的所有内容已经被删除,因此该目录此时为,`rmdir`可以成功删除它。 2. `nftw`函数的第四个参数`nopenfd`指定同时打开的文件描述符的最大数量,这里设为64(根据系统调整)。 3. 标志`FTW_DEPTH`表示深度优先遍历(即先遍历子目录再处理当前目录),这样保证在删除目录时,目录已经为。`FTW_PHYS`表示不跟随符号链接(即处理符号链接本身,而不是指向的目标)。 注意:在编译时,需要定义`_XOPEN_SOURCE`宏(如代码开头所示)以启用nftw函数。编译命令如下(以MIPS交叉编译为例): ```bash mips-linux-gnu-gcc-7.2.0 -EL -o delete_dir delete_dir.c -std=c99 ``` 替代方案: 如果不使用nftw,也可以自己编写递归函数来删除目录,但使用nftw更为简洁和安全。 另外,引用[3]展示了一个Java递归删除目录的例子,其思路与C语言递归删除类似:先递归删除目录和文件,再删除当前目录。 因此,在C语言删除空目录,我们可以使用nftw函数配合remove函数来实现。 引用: [^1]: Linux下C语言中创建和删除目录的函数。 [^2]: c语言删除文件夹。 [^3]: 删除空目录 delete(含代码)。 §§相关问题§§ 1. 如何在嵌入式系统中安全地使用递归删除目录? 2. 使用nftw函数删除目录时有哪些注意事项? 3. 如何在不支持nftw的系统中实现空目录删除?</think>### 在 C 语言删除空目录的完整解决方案 在 C 语言中,标准库函数 `rmdir()` 只能删除**空目录**(引用[1])。要删除包含文件和子目录空目录,需要使用递归遍历和删除的方法。以下是两种常用方法: --- #### 方法 1:使用 `nftw()` 函数(推荐) 这是 POSIX 标准方法,适合 Linux/Unix 系统: ```c #define _XOPEN_SOURCE 700 // 启用 POSIX 扩展 #include <stdio.h> #include <ftw.h> #include <unistd.h> // 删除回调函数 static int remove_path(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf) { return remove(fpath); // 删除文件或空目录 } // 删除空目录主函数 void delete_directory(const char *path) { // 使用深度优先遍历(FTW_DEPTH)删除 nftw(path, remove_path, 64, FTW_DEPTH | FTW_PHYS); } ``` **说明:** 1. `nftw()` 遍历目录树(引用[2]) 2. `FTW_DEPTH` 确保子目录先于父目录被处理 3. `remove()` 函数自动处理文件和空目录 4. 编译时需启用 POSIX 扩展:`gcc -std=c99 -D_XOPEN_SOURCE=700 ...` --- #### 方法 2:手动递归实现(跨平台) 适用于需要更多控制或特殊处理的场景: ```c #include <dirent.h> #include <unistd.h> #include <string.h> void delete_directory_recursive(const char *path) { DIR *dir = opendir(path); if (!dir) return; struct dirent *entry; while ((entry = readdir(dir)) != NULL) { if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0) continue; char subpath[1024]; snprintf(subpath, sizeof(subpath), "%s/%s", path, entry->d_name); if (entry->d_type == DT_DIR) { delete_directory_recursive(subpath); // 递归删除目录 } else { unlink(subpath); // 删除文件 } } closedir(dir); rmdir(path); // 删除已清目录 } ``` --- ### 使用示例 ```c int main() { const char *dir_path = "/path/to/non-empty/directory"; // 方法1 delete_directory(dir_path); // 方法2 delete_directory_recursive(dir_path); return 0; } ``` ### 重要注意事项 1. **权限要求**:需要目录及其内容的写权限 2. **符号链接**:`FTW_PHYS` 标志确保不跟随符号链接 3. **错误处理**:实际代码中应添加错误检查 4. **安全考虑**:避免路径遍历攻击,验证路径 5. **资源限制**:深度递归可能耗尽栈间(方法2) > **嵌入式系统提示**:在资源受限环境中,优先使用方法1(`nftw()`),它使用堆内存而栈内存,避免栈溢出风险[^1][^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值