免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 3739 | 回复: 2
打印 上一主题 下一主题

linux/unix 目录操作系统调用 (最后的小测试程序有bug) [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-09-10 17:23 |只看该作者 |倒序浏览
本帖最后由 j3kljs02398j 于 2011-09-10 17:33 编辑

TABLE
1. 目录操作相关的系统调用
        1.1 mkdir和rmdir系统调用
        1.1.1 实例
        1.2 chdir, getcwd系统调用
        1.2.1 实例
        1.3 opendir, closedir, readdir,
        1.3.1 实例:递归便利目录



1. 目录操作相关的系统调用

        1.1 mkdir和rmdir系统调用

  1. filename: mk_rm_dir.c
  2. #include <sys/stat.h>
  3. int mkdir(const char *path, mode_t mode);
  4. return:
  5.         S        0
  6.         F        -1
  7. note:
  8.         mode权限至少要有执行权限。
复制代码

  1. #include <unistd.h>
  2. int rmdir(const char *pathname);
  3. return:
  4.         S        0
  5.         F        -1
  6. note:
  7.         pathname目录必须是空目录。
复制代码
1.1.1 实例

  1. #include <unistd.h>
  2. #include <sys/stat.h>
  3. #include <stdio.h>
  4. #include <assert.h>

  5. #define MODE        (S_IRUSR | S_IWUSR | S_IXUSR | S_IXGRP | S_IXOTH)

  6. int main(int argc, char *argv[])
  7. {
  8.         char        *pname;

  9.         assert(argc == 2);
  10.         pname = argv[1];
  11.         assert(mkdir(pname, MODE) == 0);
  12.         printf("create %s successful!\n", pname);
  13.         assert(rmdir(pname) == 0);
  14.         printf("rm %s\n", pname);
  15.        
  16.         return 0;
  17. }
复制代码
测试:

  1. [qtlldr@qtldr editing]$ ./mk_rm_dir  testdir
  2. create testdir successful!
  3. rm testdir
  4. [qtlldr@qtldr editing]$
复制代码
1.2 chdir, getcwd系统调用

  1. #include <unistd.h>
  2. int chdir(const char *pathname);
  3. return:
  4.         S        0
  5.         F        -1
复制代码

  1. #include <unistd.h>
  2. char *getpwd(char *buf, size_t size);
  3. return:
  4.         S        buf
  5.         F        NULL
  6. note:
  7.         buf是缓冲地址,size是buf的长度。该缓冲必须有足够的长度以容纳绝对路径名加上一个null终止符。
复制代码
1.2.1 实例

  1. filename:ch_get_dir.c
  2. #include <unistd.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. #include <assert.h>

  6. #define BUFSIZE        (50)

  7. int main(void)
  8. {
  9.         char buf[BUFSIZE];

  10.         memset((void *)buf, '\0', sizeof buf);

  11.         assert(chdir("/tmp") == 0);
  12.         printf("chdir to /tmp successful\n");
  13.         assert(getcwd(buf, BUFSIZE) != NULL);
  14.         printf("now the directory is %s\n", buf);

  15.         return 0;
  16. }
复制代码
测试:

  1. [qtlldr@qtldr editing]$ ./ch_get_dir
  2. chdir to /tmp successful
  3. now the directory is /tmp
  4. [qtlldr@qtldr editing]$
复制代码
1.3 opendir, closedir, readdir,

  1. #include <sys/type.s>
  2. #include <dirent.h>

  3. DIR *opendir(const char *dirname);
  4. return:
  5.         S        DIR指针
  6.         F        NULL
  7. note:
  8.         DIR是一种目录结构,类似FILE。
复制代码

  1. #include <sys/types.h>
  2. #include <dirent.h>

  3. struct dirent *readir(DIR *dirp);
  4. return:
  5.         S        一个指向保存目录流下一个目录项的dirent指针
  6.         F        NULL
  7. note:
  8.         struct dirent {
  9.                 char        d_name[NAME + 1]; /* \0结尾的文件名 */
  10.         }
  11.         到达目录尾或出错返回NULL,但是到达目录尾不会设置errno,出错则设置。
  12.         如果在readir的同时有其他进程在目录中创建或者删除文件爱你,readdir不保证能列处该目录中所有文件。
复制代码

  1. #include <sys/types.h>
  2. #include <dirent.h>
  3. int closedir(DIR *dirp);
  4. return:
  5.         S        0
  6.         F        -1
复制代码
1.3.1 实例:递归便利目录

  1. filename:help.txt 帮助文档


  2.                         本程序只为学习linux目录操作而写

  3. printdir
  4.         输出目录文件或者统计目录中的文件数目

  5. 语法:
  6.         printdir [option] <files...>
  7. 选项:
  8.         -l
  9.                 输出目录下的文件名
  10.         -c
  11.                 统计目录下的文件
  12.         -d n
  13.                 指定最大层次,最大为30
  14. 默认行为:
  15.         如果没有指定选项,那么只输出该目录下的文件名
  16.        
  17. BUG:
  18.         -l与 -c选项不能同时使用,如果同时使用统计出错。(以后会修正)
  19. 反馈:
  20.         qtlldr@gmail.com

  21. TIME:
  22. Sat Sep 10 16:58:10 CST 2011

  23.                         本程序只为学习linux目录操作而写
复制代码

  1. filename:printdir.c

  2. #include <unistd.h>
  3. #include <fcntl.h>
  4. #include <sys/stat.h>
  5. #include <sys/types.h>
  6. #include <dirent.h>
  7. #include <stdio.h>
  8. #include <string.h>
  9. #include <stdlib.h>

  10. #define INDENT_DEPTH        (4) /* 列举文件时的缩进数 */
  11. #define DEPTH_MAX (30)        /* 递归便利的最大层次 */
  12. #define HELPFILE ("help.txt")

  13. typedef int count_t;

  14. struct nfiletype {
  15.         count_t ndir;
  16.         count_t nreg;
  17.         count_t nchr;
  18.         count_t        nfifo;
  19.         count_t        nsock;
  20.         count_t nchar;
  21.         count_t nblock;
  22.         count_t        nlink;
  23.         count_t ntotol;
  24.         count_t nunknow;
  25. };/*记录各个类型文件的数目*/

  26. int        DEPTH = 20; /* 递归层级限制 */
  27. int idepth_count = 1;
  28. int idepth_print = 1;
  29. static struct nfiletype
  30. *count_files(const char *pathname,
  31.                         struct nfiletype *nfile);

  32. static void
  33. printdir(const char *pathname, int indent);



  34. int main(int argc, char **argv)
  35. {
  36.         int opt;
  37.         int        depth_opt;
  38.         int        count_flag = 0;
  39.         int        print_flag = 0;
  40.         char *parg = NULL;
  41.         struct nfiletype nfiles = {0};

  42.         int        fd_help;
  43.         char buf_help[BUFSIZ];
  44.         int nread_help;
  45.         char *filename_help = HELPFILE;

  46.         while ((opt = getopt(argc, argv, "lhd:c")) != -1) {
  47.                 switch (opt) {
  48.                         case 'l':
  49.                                 print_flag = 1;
  50.                                 break;
  51.                         case 'c':
  52.                                 count_flag = 1;
  53.                                 break;
  54.                         case 'd':
  55.                                 depth_opt = strtol(optarg, NULL, 10);
  56.                                 DEPTH = depth_opt <= DEPTH_MAX ? depth_opt: DEPTH;
  57.                                 break;
  58.                         case ':':
  59.                                 printf("option needs a value\n");
  60.                                 break;
  61.                         case '?':
  62.                                 printf("unknown option :%c\n", optopt);
  63.                                 break;
  64.                         case 'h':
  65.                                 fd_help = open(filename_help, O_RDONLY);
  66.                                 if (fd_help != -1) {
  67.                                         while ((nread_help = read(fd_help, buf_help, BUFSIZ)) > 0) {
  68.                                                 write(1, buf_help, nread_help);
  69.                                         }
  70.                                         close(fd_help);
  71.                                 } else {
  72.                                         fprintf(stderr, "open %s failed!\n", filename_help);
  73.                                 }
  74.                                 return 0;
  75.                 }
  76.         }
  77.         /* 如果没有选项,那么默认是打印目录 */
  78.         if (!print_flag && !count_flag)
  79.                 print_flag = 1;

  80.         for( ; optind < argc; optind++) {
  81.                 parg = argv[optind];
  82.                 if (print_flag) {
  83.                         //printf("DEBUG-- printdir --%s\n", parg);
  84.                         printdir(parg, 4);
  85.                 }
  86.                 if (count_flag) {
  87.                         memset((void *)&nfiles, '\0', sizeof nfiles);
  88.                         //printf("DEBUG-- count_files--%s\n", parg);
  89.                         count_files(parg, &nfiles);
  90.                         printf("In the %s there are :\n", parg);
  91.                         printf("         directory %d\n", nfiles.ndir);
  92.                         printf("         regular file %d\n", nfiles.nreg);
  93.                         printf("         specal character file %d\n", nfiles.nchr);
  94.                         printf("         special block file %d\n", nfiles.nblock);
  95.                         printf("         fifo file %d\n", nfiles.nfifo);
  96.                         printf("         sock file %d\n", nfiles.nsock);
  97.                         printf("         link file %d\n", nfiles.nlink);
  98.                         printf("         unknown file %d\n", nfiles.nunknow);
  99.                         printf("Total %d\n", nfiles.ntotol);
  100.                 }
  101.         }
  102.         return 0;
  103. }
  104. /*
  105. *function: 对该目录下的文件类型进行统计
  106. * input arg:
  107. *                pathname:目录名指针
  108. *                nfile:记录文件类型数目的结构体指针
  109. * return:
  110. *                记录文件类型数目的结构体指针
  111. */
  112. static struct nfiletype
  113. *count_files(const char *pathname,
  114.                         struct nfiletype *nfile)
  115. {
  116.         DIR *dp;
  117.         struct dirent *entry;
  118.         struct stat        statbuf;

  119.         //printf("DEBUG-- in count_files -- %s\n", pathname);
  120.         /* 层次控制 */       
  121.         if (idepth_count > DEPTH)
  122.                 return NULL;
  123.         idepth_count++;

  124.         if ((dp = opendir(pathname)) == NULL) {
  125.                 fprintf(stderr, "can not open %s\n", pathname);
  126.                 return NULL;
  127.         }
  128.         chdir(pathname);
  129.         while ((entry = readdir(dp)) != NULL) {
  130.                 /* 跳过 . 和 .. */
  131.                 if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
  132.                         continue;
  133.                 /* 取得文件信息 */
  134.                 if (lstat(entry->d_name, &statbuf) == -1) {
  135.                         fprintf(stderr, "can not test the %s's type\n", entry->d_name);
  136.                         return NULL;
  137.                 }
  138.                 /* 统计文件数目 */
  139.                 if (S_ISDIR(statbuf.st_mode)) { /* 是目录就递归吧 */
  140.                         //printf("DEBUG -- directory %s\n", entry->d_name);
  141.                         count_files(entry->d_name, nfile);
  142.                         nfile->ndir++;
  143.                 }
  144.                 else if (S_ISREG(statbuf.st_mode)) {
  145.                         //printf("DEBUG -- regular file %s\n", entry->d_name);
  146.                         nfile->nreg++;
  147.                 }
  148.                 else if (S_ISCHR(statbuf.st_mode))
  149.                         nfile->nchr++;
  150.                 else if (S_ISBLK(statbuf.st_mode))
  151.                         nfile->nblock++;
  152.                 else if (S_ISLNK(statbuf.st_mode))
  153.                         nfile->nlink++;
  154.                 else if (S_ISFIFO(statbuf.st_mode))
  155.                         nfile->nfifo++;
  156.                 else if (S_ISSOCK(statbuf.st_mode))
  157.                         nfile->nsock++;
  158.                 else nfile->nunknow++;
  159.                 nfile->ntotol++;
  160.         }
  161.         chdir("..");
  162.         closedir(dp);
  163.         return nfile;
  164. }

  165. /*
  166. nblock; *function:列出目录中的文件
  167. nlink;  *input arg:
  168. ntotol; *                pathname: 目录名
  169. *return:
  170. *                void
  171. */
  172. static void
  173. printdir(const char *pathname, int indent)
  174. {
  175.         DIR *dp;
  176.         struct dirent *entry;
  177.         struct stat        statbuf;

  178.         /* 层次控制 */       
  179.         if (idepth_print > DEPTH)
  180.                 return ;
  181.         idepth_print++;

  182.         if ((dp = opendir(pathname)) == NULL) {
  183.                 fprintf(stderr, "can not open %s\n", pathname);
  184.                 return ;
  185.         }
  186.         chdir(pathname);
  187.         while ((entry = readdir(dp)) != NULL) {
  188.                 /* 跳过 . 和 .. */
  189.                 if (strcmp(entry->d_name, ".") == 0 || strcmp(entry->d_name, "..") == 0)
  190.                         continue;
  191.                 if (lstat(entry->d_name, &statbuf) == -1) {
  192.                         fprintf(stderr, "can not test the %s's type\n", entry->d_name);
  193.                         return ;
  194.                 }
  195.                 if (S_ISDIR(statbuf.st_mode)) { /* 是目录就递归吧 */
  196.                         printf("%*s%s/\n", indent," ",  entry->d_name);
  197.                         printdir(entry->d_name, indent + INDENT_DEPTH);
  198.                 }
  199.                 else {
  200.                         printf("%*s%s\n", indent," ", entry->d_name);
  201.                         }
  202.         }
  203.         chdir("..");
  204.         closedir(dp);
  205. }
复制代码

论坛徽章:
0
2 [报告]
发表于 2011-09-12 21:33 |只看该作者
lz,贴出来让大家学习的吗?

论坛徽章:
0
3 [报告]
发表于 2011-09-15 14:56 |只看该作者
回复 2# dengcqw


    自我娱乐
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP