主要用到的函数可以参考头文件,仅仅支持ls -l这功能,扩展就交给大家了0.0

相关测试图片:

编译 gcc -std=c99 ls_l.c -o ls

运行 ./ls -l

( 请勿在文件结构复杂的目录下执行,程序会挂的!)

话不多说,直接上码

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <unistd.h>
  4. #include <sys/stat.h>
  5. #include <sys/types.h>
  6. #include <string.h>
  7. #include <time.h>
  8. #include <pwd.h>
  9. #include <grp.h>
  10. #include <time.h>
  11. #include <dirent.h>
  12.  
  13. #define MAX_FILE_NUM 200
  14.  
  15. //可能还有一些小问题没有解决,功能基本已经实现,如有建议,望大佬赐教
  16.  
  17. typedef struct LS
  18. {
  19. char mode[]; // 文件的模式
  20. int dir_num; // 是否目录或目录中包含目录的数量
  21. char user[]; // 文件的用户名
  22. char group[]; // 文件的组名
  23. long size; // 文件的字节数
  24. char time[]; // 文件的最后修改时间
  25. char year[]; // 拓展用,年份
  26. char mon[]; // 月份
  27. char hour[]; // 时
  28. char min[]; // 分
  29. int st_mode; // 文件类型和权限
  30. char name[]; // 文件名
  31. }LS;
  32.  
  33. // 获取文件的模式
  34. char* file_mode(mode_t m,char* str)
  35. {
  36. if(S_ISREG(m))
  37. str[] = '-';
  38. else if(S_ISDIR(m))
  39. str[] = 'd';
  40. else if(S_ISCHR(m))
  41. str[] = 'c';
  42. else if(S_ISBLK(m))
  43. str[] = 'b';
  44. else if(S_ISFIFO(m))
  45. str[] = 'q';
  46. else if(S_ISLNK(m))
  47. str[] = 'l';
  48. // else if(S_ISSOCK(m))
  49. // str[0] = 's';
  50. else
  51. str[] = '?';
  52.  
  53. str[] = '\0';
  54.  
  55. strcat(str,S_IRUSR&m?"r":"-");
  56. strcat(str,S_IWUSR&m?"w":"-");
  57. strcat(str,S_IXUSR&m?"x":"-");
  58.  
  59. strcat(str,S_IRGRP&m?"r":"-");
  60. strcat(str,S_IWGRP&m?"w":"-");
  61. strcat(str,S_IXGRP&m?"x":"-");
  62.  
  63. strcat(str,S_IROTH&m?"r":"-");
  64. strcat(str,S_IWOTH&m?"w":"-");
  65. strcat(str,S_IXOTH&m?"x":"-");
  66.  
  67. return str;
  68. }
  69.  
  70. // 获取目录的数量
  71. int dir_count(char* path)
  72. {
  73. DIR *dir;
  74. dir = opendir(path);
  75. struct dirent *dirent;
  76. int count = ;
  77. while((dirent = readdir(dir)) != NULL)
  78. {
  79. if(dirent->d_type == )
  80. count++;
  81. }
  82. closedir(dir);
  83. return count;
  84. }
  85.  
  86. // 是否是目录或目录下有目录
  87. int is_dir(struct dirent *dirent)
  88. {
  89. char* a = dirent->d_name;
  90. if(dirent->d_type == )
  91. return ;
  92. if(dirent->d_type == )
  93. {
  94. if(dir_count(a) == )
  95. return ;
  96. else
  97. return dir_count(a);
  98. }
  99. }
  100.  
  101. // 获取用户名
  102. char* file_user(uid_t st_uid,char* str)
  103. {
  104. struct passwd *user;
  105. user = getpwuid(st_uid);
  106. sprintf(str,"%s",user->pw_name);
  107. return str;
  108. }
  109.  
  110. // 获取组名
  111. char* file_group(uid_t st_uid,char* str)
  112. {
  113. struct passwd *user;
  114. user = getpwuid(st_uid);
  115. struct group *grp;
  116. grp = getgrgid(user->pw_gid);
  117. sprintf(str,"%s",grp->gr_name);
  118. return str;
  119. }
  120.  
  121. // 获取文件大小
  122. off_t file_size(struct stat buf)
  123. {
  124. off_t size = buf.st_size;
  125. return size;
  126. }
  127.  
  128. // 获取最后修改时间
  129. char* file_time(time_t mt,char* str)
  130. {
  131. struct tm* t = localtime(&mt);
  132. sprintf(str,"%d月 %02d %02d:%02d",t->tm_mon+,t->tm_mday,t->tm_hour,t->tm_min);
  133. return str;
  134. }
  135.  
  136. // 获取文件的数量
  137. int file_count(char* path)
  138. {
  139. DIR *dir;
  140. dir = opendir(path);
  141. struct dirent *dirent;
  142. int count = ;
  143. while((dirent = readdir(dir)) != NULL)
  144. {
  145. count++;
  146. }
  147. closedir(dir);
  148. return count;
  149. }
  150.  
  151. // 交换
  152. void equal(LS *a,LS *b)
  153. {
  154. strcpy(a->mode,b->mode);
  155. a->dir_num = b->dir_num;
  156. strcpy(a->user,b->user);
  157. strcpy(a->group,b->group);
  158. a->size = b->size;
  159. strcpy(a->time,b->time);
  160. a->st_mode = b->st_mode;
  161. strcpy(a->name,b->name);
  162. }
  163.  
  164. // 排序
  165. void sort(LS *info,int index)
  166. {
  167. LS *temp = (LS*)malloc(sizeof(LS));
  168. for(int i=index-; i>; i--)
  169. {
  170. for(int j=; j<i; j++)
  171. {
  172. if(strcmp(info[i].name,info[j].name)<)
  173. {
  174. equal(temp,&info[i]);
  175. equal(&info[i],&info[j]);
  176. equal(&info[j],temp);
  177. }
  178. }
  179. }
  180. }
  181.  
  182. // 输出结构体
  183. void show_ls(LS *info,int index)
  184. {
  185. for(int i=; i<index; i++)
  186. {
  187. //printf("%d: ",i);
  188. printf("%s \033[0m",info[i].mode);
  189. printf("%d ",info[i].dir_num);
  190. printf("%s ",info[i].user);
  191. printf("%s ",info[i].group);
  192. printf("%5ld ",info[i].size);
  193. printf(" %s ",info[i].time);
  194. //printf("%d ",info[i].st_mode);
  195. if( == info[i].st_mode)
  196. {
  197. // 颜色
  198. printf("\033[34m\033[1m%s\033[0m",info[i].name);
  199. }
  200. else if( == info[i].st_mode)
  201. {
  202. printf("\033[32m\033[1m%s\033[0m",info[i].name);
  203. }
  204. else
  205. {
  206. printf("%s",info[i].name);
  207. }
  208. if(i < index)
  209. printf("\n");
  210. }
  211. //printf("循环结束\n");
  212. }
  213.  
  214. // 创建结构体,赋值
  215. LS *create(struct stat buf,struct dirent *dirent)
  216. {
  217. LS* info = (LS*)malloc(sizeof(LS));
  218. char str[] = {};
  219. //puts(file_mode(buf.st_mode,str));
  220. strcpy(info->mode,file_mode(buf.st_mode,str));
  221. //puts(info->mode);
  222. info->dir_num = is_dir(dirent);
  223. strcpy(info->user,file_user(buf.st_uid,str));
  224. strcpy(info->group,file_group(buf.st_uid,str));
  225. info->size = file_size(buf);
  226. strcpy(info->time,file_time(buf.st_mtime,str));
  227. info->st_mode = buf.st_mode;
  228. strcpy(info->name,dirent->d_name);
  229.  
  230. return info;
  231. }
  232.  
  233. int main(int argc,char* argv[])
  234. {
  235. LS info[MAX_FILE_NUM];
  236. char* l = "-l";
  237. if(argc != )
  238. {
  239. printf("仅支持传入 -l\n");
  240. return ;
  241. }
  242. if(strcmp(argv[],l) != )
  243. {
  244. printf("\"ls:无法识别的选项\"%s\"\n",argv[]);
  245. printf("请尝试执行\"ls --help\"来获取更多信息。\n");
  246. return ;
  247. }
  248. char* a = ".";
  249. char* b = "..";
  250. char* path = malloc();
  251. strcpy(path,"./"); // 只支持当前路径
  252. int count = file_count(path);
  253.  
  254. DIR *dir;
  255. dir = opendir(path);
  256. struct dirent *dirent;
  257. int index = ; // 结构体下标
  258. int blocks = ;
  259. for(int i=; i<count; i++)
  260. {
  261. dirent = readdir(dir);
  262. struct stat buf = {};
  263. if(stat(dirent->d_name,&buf))
  264. {
  265. perror("stat");
  266. return -;
  267. }
  268.  
  269. // 跳过特殊情况
  270. if(strcmp(dirent->d_name,a)== || strcmp(dirent->d_name,b)==)
  271. continue;
  272. blocks += buf.st_blocks;
  273. //printf("%d\n",blocks);
  274. info[index++] = *create(buf,dirent);
  275. }
  276. closedir(dir);
  277. //printf("文件总数:%d\n",index);
  278. //show_ls(info,index);
  279.  
  280. printf("总用量 %d\n",blocks/);
  281. sort(info,index);
  282. show_ls(info,index);
  283. return ;
  284. }

高仿linux下的ls -l命令——C语言实现的更多相关文章

  1. 实现Linux下的ls -l命令

    基本实现了Linux下的ls -l命令,对于不同的文件显示不同的颜色和显示符号链接暂时没有实现: /************************************************** ...

  2. 模拟linux下的ls -l命令

    #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h&g ...

  3. 编程实现Linux下的ls -l

    头文件 #ifndef __FUNC_H__ #define __FUNC_H__ #include <stdio.h> #include <stdlib.h> #includ ...

  4. linux下ls -l命令(即ll命令)查看文件的显示结果分析

    在linux下使用“ls -l”或者“ls -al”或者“ll”命令查看文件及目录详情时,shell中会显示出好几列的信息.平时也没怎么注意过,今天忽然心血来潮想了解一下,于是整理了这篇博客,以供参考 ...

  5. linux下如何使用sftp命令【转】

    linux下如何使用sftp命令 from:   http://www.cnblogs.com/chen1987lei/archive/2010/11/26/1888391.html sftp 是一个 ...

  6. linux下find和grep命令详解

    在linux下面工作,有些命令能够大大提高效率.本文就向大家介绍find.grep命令,他哥俩可以算是必会的linux命令,我几乎每天都要用到他们.本文结构如下: find命令 find命令的一般形式 ...

  7. 终端的乐趣--Linux下有趣的终端命令或者工具【转】

    转自:https://blog.csdn.net/gatieme/article/details/52144603 版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原 ...

  8. Linux下的一些常用命令(一)

    在Linux环境下敲各种命令是再正常不过了,尤其是现在大多少服务器均为Linux系统,但是我又记不住这么多命令,只是偶尔在项目做完发布到服务器上的时候会涉及到,所以在网上找了一些命令,在此记录一下~ ...

  9. 实现linux下的ls

    实现linux下的ls ls的使用 ls -a 列出文件下所有的文件,包括以"."开头的隐藏文件(linux下文件隐藏文件是以.开头的,如果存在..代表存在着父目录). ls -l ...

随机推荐

  1. MongoDB 红宝书-MongoDB官网使用指南

    本文转载自Mongodb中文社区:http://www.mongoing.com/archives/27359 无论你是MongoDB的使用者.爱好者.初学者还是路人甲,有一个学习与进修的资源宝藏是千 ...

  2. WinDbg常用命令系列---.cmdtree

    .cmdtree 简介 使用形式 .cmdtree cmdfile 参数 cmdfile命令文件,包含多个你需要的命令.必须是一个文本档 使用步骤 1.使用命令创建文本文件test.wl,使用以下示例 ...

  3. WinDbg常用命令系列---线程相关操作~*

    ~ (Thread Status) 波浪符(~)命令显示指定线程或当前进程中所有线程的状态. ~ Thread 参数: Thread指定要显示的线程.如果省略此参数,将显示所有线程. 环境: 模式 仅 ...

  4. ServiceStack.OrmLite 基本操作

    原文:https://www.cnblogs.com/wang2650/category/780821.html 原文:https://www.cnblogs.com/xxfcz/p/7045808. ...

  5. JMeter学习1

    Jmeter的组织方式相对比较扁平,直接是TestPlan(相当于Project),TestPlan下创建的ThreadsGroup(相当于TestCase), Jmeter一个TestPlan也是一 ...

  6. sublime text 3插件改造之AutoFileName去掉.vue文件中img标签后面的width和height,完全去掉!!

    在.vue文件中img标签使用autofilename提示引入文件时,会在文件后面插入宽度高度,如下图: 文件后面会自动插入height和width,其实这两玩意儿在大多数时候并没卵用,然后就开始了百 ...

  7. c博客作业-我的第一篇博客

    1.你对网络专业或者计算机专业了解是怎样的? 以前接触计算机,只是把它当作娱乐的工具,并没有太过了解,现在我通过查阅了解了一些计算机的知识. 计算机专业的学生要学习的不仅是会使用,而且要学习计算机的基 ...

  8. Java两个数的和

    给你一个整数数组, 返回两个数的指数,使他们加起来等于给定的数. 你可以假设只有一个解, 并且相同的元素不会用两次. 比如: 给定数组 = [2, 7, 11, 15], 目标数 = 9, 因为[0] ...

  9. 【pytorch 代码】pytorch 网络结构可视化

    部分内容转载自 http://blog.csdn.net/GYGuo95/article/details/78821617,在此表示由衷感谢. 此方法需要安装python-graphviz:  con ...

  10. Service Function Chaining Resource Allocation: A Survey

    摘要: 服务功能链(SFC)是未来Internet的一项关键技术. 它旨在克服当前部署模型的僵化和静态限制. 该技术的应用依赖于可以将SFC最佳映射到衬底网络的算法. 这类算法称为"服务功能 ...