如何在使用dirent.h阅读时跳过目录

时间:2012-04-30 23:55:37

标签: c++ c opendir dirent.h

我正在尝试使用 dirent.h

中提供的功能递归打开文件

我的问题是: 我无法跳过无法打开的目录。我希望它打开它可以的目录,并跳过它不能的那些并移动到下一个目录而不是退出失败。 我该怎么做才能解决这个问题?

这是我尝试使用的简单代码

int acessdirs(const char *path)
{
    struct dirent *entry;
    DIR *dp;
    char fpath[300];
    if(dp=opendir(path))
    {
        while((entry=readdir(dp)))
            do things here
    }
    else
    {
        std::cout<<"error opening directory";
        return 0;
    }
    return 1;
}

我在windows 7上使用了相同的样式并且它运行正常。但它在windows xp上崩溃,当我调试它时,我发现它在尝试打开"system volume information"时崩溃了。 我真的不需要访问这个文件夹,我希望是否有任何方法可以跳过它。

这是我的真实代码:

有点长。

int listdir(const char *path) 
{
  struct dirent *entry;
  DIR *dp;

  if(dp = opendir(path))
  {
    struct stat buf ;

    while((entry = readdir(dp)))
    {
        std::string p(path);
        p += "\\";
        p += entry->d_name;
         char fpath[300];
        if(!stat(p.c_str(), &buf))
        {
            if(S_ISREG(buf.st_mode))
            {  
                sprintf(fpath,"%s\\%s",path,entry->d_name);
                stat(fpath, &buf);
                std::cout<<"\n Size of \t"<<fpath<<"\t"<<buf.st_size;
                fmd5=MDFile (fpath);        
            }//inner second if
            if(S_ISDIR(buf.st_mode) &&  
         // the following is to ensure we do not dive into directories "." and ".."
                      strcmp(entry->d_name, ".")  && strcmp(entry->d_name, "..") )
            {
                listdir(p.c_str());
            }
        }//inner first if
        else
            std::cout << "ERROR in stat\n";
    }//end while
    closedir(dp);
  }//first if
  else
  {
      std::cout << "ERROR in opendir\n";
      return 0;
  }
  return 1;   

 }//listdir()

1 个答案:

答案 0 :(得分:1)

你最大的问题似乎就在这里:

sprintf(fpath,"%s\\%s",path,entry->d_name);
stat(fpath, &buf);

没有看到fpath的声明,很难说出来,但你要么

  • fpath调用中溢出sprintf,导致未定义的行为。 “系统卷信息”是一个很长的名称。你应该真的使用snprintf

  • 不检查stat来电的返回值。如果它返回-1,我不确定buf的内容是什么。

更重要的是,如果你可以使用POSIX东西,函数ftw是标准的,并且应该提供你试图在这里实现的大部分功能。