0

我的程序遍历单个目录(非递归)并将该目录中所有文件的名称存储在一个数组中。然后,它在我的程序的第二部分使用该数组并返回有关每个文件的一些信息。我可以遍历目录,并且可以处理单个文件,但是在组合程序的两个部分时遇到了麻烦。这是我的代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <dirent.h>

int getArraySize(char* arr[]);
int getArraySize(char* arr[]) {
  return sizeof(&arr);
}

char *filesArray[200];
int main (int argc, char* argv[])
{
  DIR *dir;
  struct dirent *ent;
  int filesCtr = 0;
  if ((dir = opendir ("/home/dshah/Documents/CECS 420/Project 3")) != NULL) {
    while ((ent = readdir (dir)) != NULL) {     /* print all the files and directories within directory */
      if (strcmp(ent->d_name, ".") == 0) {
        continue;
      } else if (strcmp(ent->d_name, "..") == 0) {
        continue;
      } else if (ent->d_type == 4) { // if a directory
        continue;
      } else {
        filesArray[filesCtr] = ent->d_name;
        printf("%s\n", filesArray[filesCtr]);
        filesCtr++;
      }
    }
    closedir (dir);
  } else {     /* could not open directory */
    perror ("Could not open directory");
  }

  int i;
  for (i = 0; i < getArraySize(filesArray); i++) {
    char* filename = filesArray[i];
    FILE *file = fopen (filename, "r");
    if (file != NULL) {
      char line [128]; /* or other suitable maximum line size */
      int ctr = 1;
      while (fgets(line, sizeof line, file) != NULL) { /* read a line */
        if (strstr(line, "is") != NULL) {
          printf("%s:%d:%s", filename, ctr, line);
        }
        ctr++;
      }
      fclose (file);
    } else {
      perror (filename); /* why didn't the file open? */
    }
  }
  return 0;
}

我遇到的问题是:

char* filename = filesArray[i];

这行代码正确吗?当我设置filename为类似的字符串时它会起作用"file.txt",所以当我这样做时这不应该也起作用printf("n %s\n", filesArray[i]);吗?在这filesArray[i]行代码中是一个字符串吗?


编辑:

谢谢,这解决了问题。另一个快速的问题:我正在尝试将完整路径附加到

FILE *file = fopen (filename, "r");`

通过将其更改为

FILE *file = fopen (strcat("/home/dshah/Documents/CECS 420/Project 3/", filename), "r"); 

但它给了我一个分段错误。这不应该因为我只是指定路径吗?

4

1 回答 1

1

当你将一个数组传递给一个函数时,它会衰减为一个指针,所以当你这样做时,&arr你实际上得到了一个指向该指针的指针,并且指针的大小很可能不是原始数组的大小。如果(我的意思是如果)数组实际上是一个字符串,您可以使用它strlen来获取字符串的长度(不包括字符串终止符)。

在您的情况下,您实际上并不需要该函数,因为您已经有一个计数器告诉您数组getArraySize中有多少个字符串:变量。filesArrayfilesCtr


此外,当使用诸如返回条目readdird_name字段之类的函数时,实际上可能指向静态数组,因此您实际上不能只复制指针,而必须复制完整的字符串。这是通过以下strdup功能完成的:

filesArray[filesCtr] = strdup(ent->d_name);

请记住,完成后您必须使用free此字符串。


哦,避免在代码中使用“幻数”,例如在检查目录条目是否为子目录 ( ent->d_type == 4) 时。使用可用的宏 ( end->d_type == DT_DIR)。


最后一点,条目的d_name字段readdir只包含实际的文件名,而不是完整路径。因此,如果您想要完整路径,则必须附加路径和文件名。

于 2013-11-10T16:05:21.063 回答