如何正确使用 realpath() 读取目录中的多个文件?

How to properly use realpath() for reading multiple files within a directory?

这应该是一个读取目录中的多个文件并获取给定单词在这些文件中出现的频率的程序,我知道这部分目前还不完整。

fopen() 将 null 返回到存在且具有 777.

权限的文件

这里是来自终端的权限:

ra@ra-VirtualBox:~/Desktop/lab2/folder$ ls -l
total 12
-rwxrwxrwx 1 ra ra 21 mar 14 23:20 file1.txt
-rwxrwxrwx 1 ra ra 21 mar 14 23:20 file2.txt
-rwxrwxrwx 1 ra ra 21 mar 14 23:20 file3.txt

这里是按照this answer

的建议使用errno后得到的输出

印有 buf

READING FILE: /home/ra/Desktop/lab2/file3.txt

fopen: No such file or directory
READING FILE: /home/ra/Desktop/lab2/file1.txt

fopen: No such file or directory
READING FILE: /home/ra/Desktop/lab2/file2.txt

fopen: No such file or directory

编辑:从输出中我注意到 realpath() 在转到文件之前以某种方式删除了整个子目录。 /folder/ 不存在并且 /home/ra/Desktop/lab2/file*.txt 不存在!

而代码如下:

#include <stdio.h>

#include <stdlib.h>

#include <string.h>

#include <sys/types.h>

#include <dirent.h>

#include <unistd.h>

#include <pthread.h>

#include <limits.h>       //For PATH_MAX

typedef struct {
  FILE * fptr;
  char * word;

}
inputFile;

void * readFrequenciesFromFile(void * path) {

  static int count = 0;

}

int main(int argc, char * argv) {

  int i;

  char buf[PATH_MAX + 1];
  char * wordToSearch = "test";

  DIR * dir;
  FILE * entry_file;
  struct dirent * in_file;

  // notice that the full directory is used here
  // but buf somehow does magic and tells me the files
  // are in /lab2/ directory
  dir = opendir("/home/ra/Desktop/lab2/folder");

  if (dir == NULL) {
    printf("Error! Unable to read directory");
    exit(1);
  }

  while ((in_file = readdir(dir)) != NULL) {

    if (!strcmp(in_file -> d_name, "."))
      continue;
    if (!strcmp(in_file -> d_name, ".."))
      continue;

    realpath(in_file -> d_name, buf);
    entry_file = fopen(buf, "r");
    // printf("READING FILE: %s\n", buf);
    if (!entry_file) perror("fopen");
    if (entry_file != NULL) {
      pthread_t tid;
      inputFile * args = malloc(sizeof * args);
      args -> fptr = malloc(sizeof entry_file);
      args -> word = wordToSearch;

      if (pthread_create( & tid, NULL, readFrequenciesFromFile, args) == 0) {
        printf("Creating thread for file [%s] with ID %ld\n", in_file -> d_name, tid);
      } else {
        free(args);
      }

      fclose(entry_file);
    }
  }

  closedir(dir);
  return 0;
}

我会提到我在 VirtualBox 版本 6.1.10 r138449 (Qt5.6.2) 上使用 Ubuntu 18.04

我从哪里开始解决这个问题?这是否合理,或者我错过了什么?

我看到 ra@ra-VirtualBox:~/Desktop/lab2/folder$folderREADING FILE: /home/ra/Desktop/lab2/file3.txt 没有。问题是默认目录是 ~/Desktop/lab2/realpath() 使用的就是这个目录;也许添加 chdir() 会有所帮助:

chdir("/home/ra/Desktop/lab2/folder");
dir = opendir("/home/ra/Desktop/lab2/folder");

如果您始终检查函数调用的 return 值以识别它们何时失败,您会更好地了解问题所在。在这种情况下,如果您检查了 realpath() 调用的 return 值,您可能已经知道了。

我假设您的主目录是 /home/ra,如您的代码所示。观察您要打开的文件在 /home/ra/Desktop/lab2/folder 中。您已成功打开该目录并读取其条目,这些条目给出了其中文件的 基本名称 -- 即没有任何路径组件的名称。

无论您是通过打开文件还是通过计算它们的 realpath() 来尝试解析这些文件名,您都是相对于工作目录这样做的。但是你启动程序的方式,它的工作目录是 /home/ra/Desktop/lab2,而不是 /home/ra/Desktop/lab2/folder,所以程序要求的文件确实 存在。

您的选择有

  • 运行 包含文件的文件夹中的程序
  • 形成用于打开文件的正确路径 -- 正确的绝对路径或相对于实际工作目录的正确路径。