C语言实现文件类型统计函数

#include<dirent.h>
#include<limits.h>
#include<sys/stat.h>
#include<stdio.h>
#include<unistd.h>
#include<stdlib.h>
#include<string.h>#define FTW_F        1        //标记非目录文件
#define FTW_D        2        //标记目录文件
#define FTW_DNR        3        //标记不可读目录
#define FTW_NS        4        //标记不可获得stat的文件static char *fullpath;        //保存文件的全路径
static size_t pathlen;        //保存文件的路径长度//定义处理文件的函数
typedef int Myfunc(const char *,const struct stat*,int);
static Myfunc myfunc;
static int myftw(char *,Myfunc *);
static int dopath(Myfunc *);
char *path_alloc(size_t *size_t);/*
nreg:普通文件的个数;    ndir: 目录文件的数量;    nblk:块特殊文件的数量
nchr:字符特殊文件的数量    nfifo:管道特殊文件的数量
nslink:符号连接特殊文件的数量;    nsock:套接字文件数量; ntot:总文件数量
*/
static long nreg,ndir,nblk,nchr,nfifo,nslink,nsock,ntot;int main( int argc, char *argv[])
{int ret;if(argc != 2){printf("falut command input !\n");exit(1);}//计算各类文件的个数ret = myftw(argv[1],myfunc);ntot = nreg + ndir + nblk + nchr + nfifo + nslink + nsock;//避免除以0if(ntot == 0){ntot = 1;}printf("regular files = %7ld,%5.2f %%\n",nreg,nreg*100.0 / ntot);printf("direciotn files = %7ld,%5.2f %%\n",ndir,ndir*100.0 / ntot);printf("block special = %7ld,%5.2f %%\n",nblk,nblk*100.0 / ntot);printf("char special = %7ld,%5.2f %%\n",nchr,nchr*100.0 / ntot);printf("FIFOS  = %7ld,%5.2f %%\n",nfifo,nfifo*100.0 / ntot);printf("symbolic links = %7ld,%5.2f %%\n",nslink,nslink*100.0 / ntot);printf("sockers  = %7ld,%5.2f %%\n",nsock,nsock*100.0 / ntot);}//处理pathname并保存在一个全局的字符数组中,调用dopath
static int myftw(char *pathname,Myfunc *func)
{//为保存路径的字符串数组分配空间fullpath = path_alloc(&pathlen);//如果分配内存空间不够就重新分配if(pathlen <= strlen(pathname)){pathlen = strlen(pathname) * 2;if((fullpath = realloc(fullpath,pathlen )) == NULL);printf("realloc failed\n");}//将路径名参数保存到全路径中,fullpath是全局变量,dopath函数可以调用
    strcpy(fullpath,pathname);return (dopath(func));
}//路径数组分配
char *path_alloc(size_t *size)
{char *p = NULL;if(!size)return NULL;p = malloc(256);if(p)*size = 256;else*size = 0;return p;
}//dopath用于判断是否是目录,然后根据选择情况是直接进入myfun函数取技术
//还是递归调用dopath函数
static int dopath(Myfunc *func)
{struct stat        statbuf;struct dirent    *dirp;DIR             *dp;int             ret,n;//调用lstat获取路径名的stat信息,如果不成功,调用func函数,并传递给FTW_NSif(lstat(fullpath,&statbuf) < 0)return (func(fullpath, &statbuf, FTW_NS));//查看文件stat结构的st_mode,如果不是目录,调用func函数 //并传递给FTW_F,交由myfun进一步判断文件类型 if(S_ISDIR(statbuf.st_mode) == 0)return(func(fullpath,&statbuf,FTW_F));//最后一种情况就是该路径名代表的是一个目录,此次的fun的正常情况返回0//所以执行完func后还不会返回,会继续执行funcif((ret = func(fullpath,&statbuf,FTW_D)) != 0)return(ret);//路径处理,扩充路径空间长度n = strlen(fullpath);if(n + NAME_MAX + 2 > pathlen){pathlen *= 2;if((fullpath = realloc(fullpath,pathlen)) == NULL){printf("realoc failed\n");}}fullpath[n++] = '/';fullpath[n] = 0;//处理每个目录项if((dp = opendir(fullpath)) == NULL)return (func(fullpath,&statbuf,FTW_DNR));while((dirp = readdir(dp)) != NULL){//忽略当前目录(.)和上一级目录(..)以避免进入死循环if(strcmp(dirp->d_name,".") == 0 ||     strcmp(dirp->d_name,"..") == 0 )continue;strcpy(&fullpath[n],dirp->d_name); //在“/”之后加上当前目录项的命中if((ret = dopath(func)) != 0) //然后采用新的路径名递递归的调用dopathbreak;}fullpath[n-1] = 0;//关闭目录if(closedir(dp) < 0 )printf("can't close directory %s",fullpath);return ret;}//通过stat结构的st_mode字段来判断文件的类型,并计数
static int myfunc(const char *pathname,const struct stat *statptr,int type)
{switch(type){//会与非目录情况进行处理case FTW_F:switch (statptr->st_mode & S_IFMT){case S_IFREG:    nreg++;    break;case S_IFBLK:    nblk++;    break;case S_IFCHR:    nchr++;    break;case S_IFIFO:    nfifo++;    break;case S_IFLNK:     nslink++;    break;case S_IFSOCK:    nsock++;     break;case S_IFDIR:    printf("for S_IFDIR for %s",pathname);}break;//对于目录文件进行处理case FTW_D:ndir++;break;//对于不可读目录进行处理case FTW_DNR:printf("%s dir isn't read",pathname);break;case FTW_NS:printf("%s stat error",pathname);default:printf("%d type aren't identified, path is %s",type,pathname);}return 0;
}

 

转载于:https://www.cnblogs.com/wanghao-boke/p/11608159.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/384806.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言实现多线程排序

#include <stdio.h> #include <pthread.h> #include <stdlib.h> #include <string.h>/* 声明变量 */ int array_length, file_length; int *array_master; FILE *freader;/* 用于从文件读取数据 */ int *read_file(char *fname) {freader fopen(fnam…

linux线程操作

初始化条件变量 int pthread_cond_init(pthread_cond_t *cv,pthread_cond_attr *cattr); 函数返回值&#xff1a;返回0表示成功&#xff0c;返回其他表示失败。 参数&#xff1a; pthread_cond_attr是用来设置pthread_cond_t的属性&#xff0c;当传入的值是NULL的时候表…

Linux下多线程模拟停车场停车

#include<stdio.h> #include<string.h> #include<unistd.h> #include<stdlib.h> #include<pthread.h>#define ONE_SECOND 1000000 #define RANGE 10 #define PERIOD 2 #define NUM_THREADS 4typedef struct {int *carpark; //用一个数组来模…

【C++学习之路】第一章——C++核心方法总论

1 C核心方法总论 1.1 核心思想 通过实际项目来学习编程&#xff0c;更高效掌握编程规则&#xff0c;以及明白各种语法规则的实际应用。 实验思想&#xff1a;任何C的参考资料都不可能覆盖你遇到的所有问题&#xff0c;这个时候&#xff0c;最好的办法就是&#xff0c;编辑代…

【Verilog HDL学习之路】第一章 Verilog HDL 数字设计总论

1 Verilog HDL 数字设计总论 1.1 几个重要的概念 EDA&#xff08;Electronic Design Automation&#xff09; 电子技术自动化 EDA工具 类似于软件工程中的IDE&#xff08;集成开发环境&#xff09;&#xff0c;能够使用Verilog HDL语言描述电路设计&#xff0c;并且能够通过逻…

【学会如何学习系列】从婴儿到大学——学习的本质从未改变过

从婴儿到大学——学习的本质从未改变过 从我们出生一直到现在&#xff0c;其实&#xff0c;学习的本质从来都没有改变过&#xff0c;并且&#xff0c;婴儿时期的我们&#xff0c;是学习能力最强的时候&#xff0c;随着我们不断长大&#xff0c;外界的诱惑越来越多&#xff0c;…

【汇编语言学习之路】第一章 汇编语言核心方法论

版权声明&#xff1a;本学习笔记是本人根据小甲鱼“汇编语言学习课程”和《汇编语言》&#xff08;王爽&#xff09;的书籍&#xff0c;来记录笔记的 1 汇编语言核心方法论 1.1 学习汇编语言的必要性 汇编语言与机器语言是一一对应关系&#xff0c;它的本质是机器语言的代号。…

蓝桥单片机赛题及模拟题代码

链接&#xff1a;https://pan.baidu.com/s/1BVB6VILEed0ufqRDMhvALg 提取码&#xff1a;ukx7

【Verilog HDL学习之路】第二章 Verilog HDL的设计方法学——层次建模

2 Verilog HDL的设计方法学——层次建模 重要的思想&#xff1a; 在语文教学中&#xff0c;应该先掌握核心方法论&#xff0c;再用正确的方法论去做题目&#xff0c;这样能够逐渐加深对于方法论的理解&#xff0c;做题的速度和准确率也会越来越高。在Verilog HDL中&#xff0c…

stm32机械臂资料含视频

这是在网上买的机械臂的资料 含视频及相关软件 在这里分享给大家 不过很大 但是内容很全 链接&#xff1a;https://pan.baidu.com/s/1Fd18ww8jxLH8ChqomstZtw 提取码&#xff1a;147g

【Verilog HDL】第四章 模块的端口连接规则——污水处理模型

先放上连接规则的简图&#xff0c;再详细解释 1. 构建模型——污水处理之流水模型 我们先将上述结构构件一个简单模型&#xff0c;以帮助我们理解。 污水&#xff1a;输入数据净水&#xff1a;输出数据双向数据暂不讨论&#xff0c;取输入和输出的交集即可污水处理厂&…

蓝桥杯嵌入式第七届模拟题 代码

链接&#xff1a;https://pan.baidu.com/s/1fdGC20A51axxPGpoyRL8-w 提取码&#xff1a;by4u

三级嵌入式选择知识点整理

SoC芯片 通用SoC是系统级芯片 既可以是单核 也可以是多核 该芯片中可以包含数字电路 模拟电路 数字模拟混合电路 及射频电路 片上系统可使用单个芯片进行数据的采集 转换 储存 处理 及I/O口功能 智能手机 和平板都使用的SOC WAV是未压缩的数字音频 音质与CD相当 音频视频压缩…

【Verilog HDL】命名的规则研究

Verilog命名规范参考资料 1. 什么可以被命名&#xff1f; 模块的名称模块实例的名称各种数据类型的名称 这些名称我们称之为标识符&#xff0c;标识符的命名规则不再强调&#xff0c;与C语言类似&#xff0c;字母、数字、下划线&#xff08;_&#xff09;和美元符号&#xf…

【Verilog HDL】深入理解部分语法规则的本质

1. 门级描述 统一规则&#xff1a; 门类型 (输出&#xff0c;输入); 细化规则&#xff1a; 与/或门&#xff1a; 多入一出 门 (输出&#xff0c;输入1&#xff0c;输入2,……);缓冲门/非门&#xff1a;一入多出 门 (输出1&#xff0c;输出2,……输出n&#xff0c;输入); 门…

三级嵌入式填空整理

实时 可预测性是实时系统的重要性能标准 按照响应时间 实时操作系统可分为 1.普通实时操作系统 响应时间一般是秒级 2.强实时操作系统 响应时间为毫秒和微秒级 3.弱实时操作系统 响应时间为数十秒 RTOS 响应中断请求并完成相应中断服务子程序的时间非常快 这个时间具有一致性…

【Verilog HDL】从逻辑电路图到门级建模——人工翻译的方法论

从左到右&#xff0c;从上到下 先搞定缓冲/非门&#xff0c;再写与/或门 1. 实例解读 先以四选一数据选择器进行说明 对于数字逻辑的部分不再说明&#xff0c;直接进行逻辑电路图到Verilog门级建模的人工翻译过程的描述。 1.1 端口和线网分析 确定输入/输出端口 输入端口 …

三级嵌入式 汇编指令汇总

ARM条件码 EQ 相等 NE 不相等 CS/HS 无符号大于等于 CC/LO 无符号小于 HI 无符号大于 LS 无符号小于等于 GE 带符号大于等于 L…

【Verilog HDL】语句的并发执行

1. 实践得到的启发 先从一个简单的现象得出结论&#xff0c;Verilog语句是并发执行的&#xff01; 同时&#xff0c;这也是**$monitor系统任务为全局有效**的一个重要支持因素&#xff0c;如果没有并发&#xff0c;它是完不成这项功能的实现的。 众所周知&#xff0c;高级语…

linux下 最常用基本命令

常用命令 基本命令 pwd 打印绝对路径 ls 路径 列举文件名 ls 列举文件的权限 属于哪个用户 容量大小 修改…