C - access() 表示文件存在,但 fopen() 表示不存在



我有一个试图读取文件的c程序。使用 access() 命令,它说文件在那里,但fopen()返回 NULL,errno 说文件不存在。

截断的代码片段:

  FILE *fp;
  char *filename = strdup(git_dir);
  strcat(filename, "/HEAD");
  printf(git_dir);
  printf(":");
  printf(filename);
  printf(":");
  if (access(filename, F_OK)) {
    printf("Y U NO OPEN:");
  }
  fp = fopen(filename, "r");
  if (fp == NULL) {
    printf(strerror(errno));
    return;
  }

正如您可能从代码中可以看出的那样,这是尝试打开 git 存储库的 .git/HEADS 文件。此失败的特定存储库被克隆到另一个存储库的子目录中,然后作为子模块添加。我没有通过运行 git submodule update 命令克隆的问题。

上面的代码不会打印"Y U NO OPEN:",但它确实打印出strerror()。我将文件名打印到屏幕上并用less打开文件,它打开正常。这个程序正在生成我的zsh提示符,所以我想它是由我的用户运行的,并且文件的权限是

-rw-r--r-- 1 ben users   23 Jun 30 13:32 HEAD

有什么建议吗?

你不能这样做:

char *filename = strdup(git_dir);
strcat(filename, "/HEAD");

您将字符串"/HEAD"附加到文件名,但文件名只有空间容纳git_dir的内容。因此,您正在写入缓冲区,覆盖内存,并可能导致严重破坏,并且可能发生各种不可预测的行为。

请改为执行以下操作:

char *filename = malloc(strlen(git_dir) + 6); //length of /HEAD + null terminator
strcpy(filename, git_dir);
strcat(filename, "/HEAD");

请记住,标准输出通常也是行缓冲的。这意味着,如果执行printf("Y U NO OPEN:");,则可能不会立即看到输出。打印换行符 以刷新输出。

最新更新