当前位置: 首页 > 编程笔记 >

C++利用 _findfirst与_findnext查找文件的方法

景鸿才
2023-03-14
本文向大家介绍C++利用 _findfirst与_findnext查找文件的方法,包括了C++利用 _findfirst与_findnext查找文件的方法的使用技巧和注意事项,需要的朋友参考一下

C++ 文件查找

在C++中我们要如何查找文件呢?我们需要一个结构体和几个大家可能不太熟悉的函数。这些函数和结构体在的头文件中,结构体为struct _finddata_t ,函数为_findfirst、_findnext和_fineclose。具体如何使用,下面来一起看看吧

_findfirst与_findnext查找文件

一、这两个函数均在io.h里面。

二、首先了解一下一个文件结构体:

struct _finddata_t {
 unsigned attrib;
 time_t  time_create; 
 time_t  time_access; 
 time_t  time_write;
 _fsize_t size;
 char  name[260];
};

time_t,其实就是long

而_fsize_t,就是unsigned long

现在来解释一下结构体的数据成员吧。

attrib,就是所查找文件的属性:_A_ARCH(存档)、_A_HIDDEN(隐藏)、_A_NORMAL(正常)、_A_RDONLY(只读)、 _A_SUBDIR(文件夹)、_A_SYSTEM(系统)。

time_create、time_access和time_write分别是创建文件的时间、最后一次访问文件的时间和文件最后被修改的时间。

size:文件大小

name:文件名。

三、用 _findfirst 和 _findnext 查找文件

1、_findfirst函数:long _findfirst(const char *, struct _finddata_t *);

第一个参数为文件名,可以用"*.*"来查找所有文件,也可以用"*.cpp"来查找.cpp文件。第二个参数是_finddata_t结构体指针。若查找成功,返回文件句柄,若失败,返回-1。

2、_findnext函数:int _findnext(long, struct _finddata_t *);

第一个参数为文件句柄,第二个参数同样为_finddata_t结构体指针。若查找成功,返回0,失败返回-1。

3、_findclose()函数:int _findclose(long);

只有一个参数,文件句柄。若关闭成功返回0,失败返回-1。

#include <io.h>
#include <iostream>
#include <fstream>
using namespace std;

bool transfer(string fileName, int exeNum );
void dfsFolder(string folderPath, ofstream &fout);

int main()
{
  _finddata_t file;
  int k;
  long HANDLE;
  k = HANDLE = _findfirst("*.*", &file);
  while (k != -1)
  {
    cout << file.name << endl;
    k = _findnext(HANDLE, &file);
  }
  _findclose(HANDLE);

  transfer("C:\\Windows\\*.exe", 0);
  ofstream o_fstream;

  dfsFolder("E:\\\WHU\\Study", o_fstream);


  return 0;
}

//_findfirst 函数返回的是匹配到文件的句柄,数据类型为long。
//遍历过程可以指定文件类型,这通过FileName的赋值来实现,例如要遍历C : \WINDOWS下的所有.exe文件

bool transfer(string fileName , int exeNum)
{
  _finddata_t fileInfo;
  long handle = _findfirst(fileName.c_str(), &fileInfo);

  if (handle == -1L)
  {
    cerr << "failed to transfer files" << endl;
    return false;
  }

  do
  {
    exeNum++;
    cout << fileInfo.name << endl;
  } while (_findnext(handle, &fileInfo) == 0);
  cout << " .exe files' number: " << exeNum << endl;

  return true;
}

//遍历文件夹及其子文件夹下所有文件。操作系统中文件夹目录是树状结构,使用深度搜索策略遍历所有文件。用到_A_SUBDIR属性


//在判断有无子目录的if分支中,由于系统在进入一个子目录时,匹配到的头两个文件(夹)是"."(当前目录),".."(上一层目录)。
//需要忽略掉这两种情况。当需要对遍历到的文件做处理时,在else分支中添加相应的代码就好

void dfsFolder(string folderPath, ofstream &fout)
{
  _finddata_t FileInfo;
  string strfind = folderPath + "\\*";
  long Handle = _findfirst(strfind.c_str(), &FileInfo);

  if (Handle == -1L)
  {
    cerr << "can not match the folder path" << endl;
    exit(-1);
  }
  do{
    //判断是否有子目录 
    if (FileInfo.attrib & _A_SUBDIR)
    {
      //这个语句很重要 
      if ((strcmp(FileInfo.name, ".") != 0) && (strcmp(FileInfo.name, "..") != 0))
      {
        string newPath = folderPath + "\\" + FileInfo.name;
        dfsFolder(newPath, fout);
      }
    }
    else
    {
      fout<<folderPath.c_str() << "\\" << FileInfo.name << " ";
      cout << folderPath.c_str() << "\\" << FileInfo.name << endl;
    }
  } while (_findnext(Handle, &FileInfo) == 0);

  _findclose(Handle);
  fout.close();
}


//#include <iostream>  
//#include <string>  
//#include <io.h>  
//using namespace std;
//
//int main()
//{
//  _finddata_t file;
//  long longf;
//  string tempName;
//  //_findfirst返回的是long型; long __cdecl _findfirst(const char *, struct _finddata_t *)  
//  if ((longf = _findfirst("E:\\WHU\\Study\\*.*", &file)) == -1l)
//  {
//    cout << "文件没有找到!\n";
//    return 0;
//  }
//  do
//  {
//    cout << "文件列表:\n";
//    tempName = file.name;
//    if (tempName[0] == '.')
//      continue;
//    cout << file.name<<endl;
//
//    if (file.attrib == _A_NORMAL)
//    {
//      cout << " 普通文件 ";
//    }
//    else if (file.attrib == _A_RDONLY)
//    {
//      cout << " 只读文件 ";
//    }
//    else if (file.attrib == _A_HIDDEN)
//    {
//      cout << " 隐藏文件 ";
//    }
//    else if (file.attrib == _A_SYSTEM)
//    {
//      cout << " 系统文件 ";
//    }
//    else if (file.attrib == _A_SUBDIR)
//    {
//      cout << " 子目录 ";
//    }
//    else
//    {
//      cout << " 存档文件 ";
//    }
//    cout << endl;
//  } while (_findnext(longf, &file) == 0);//int __cdecl _findnext(long, struct _finddata_t *);如果找到下个文件的名字成功的话就返回0,否则返回-1  
//
//  _findclose(longf);
//
//  return 0;
//}

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对小牛知识库的支持。

 类似资料:
  • iOS自身带有快捷查看文件方式的框架 ,可以对一些常用资源文件进行查看,能够查看的文件格式包括图片,word,ppt,pdf等。Demo是读取和查看保存在设备Document文件夹的文件。 [Code4App.com]

  • 本文向大家介绍C#实现利用Windows API读写INI文件的方法,包括了C#实现利用Windows API读写INI文件的方法的使用技巧和注意事项,需要的朋友参考一下 本文实例讲述了C#实现利用Windows API读写INI文件的方法。分享给大家供大家参考。具体如下: 写入时,如果没有INI文件,自动创建INI 如果在创建时,GetLastError:5 检查IniPath是否添加了文件名称

  • 我想你可以告诉我,这真的只是打印同一行两次,一次删除一个小写e和另一个删除一个大写e。我正在寻找一种方法来合并这两个替换,然后让它替换并打印,如果它找到一个e或只是打印“所有是好的!”如果没有“E”。

  • 本文向大家介绍利用C语言替换文件中某一行的方法,包括了利用C语言替换文件中某一行的方法的使用技巧和注意事项,需要的朋友参考一下 文件中存贮的内容如下所示: 通过使用下面的几个函数,fopen,fprintf,fscanf,fseek,ftell 。 具体的函数函数原型如下所示: 首先,数据写入到文件中的是追加的形式。 具体的写入代码如下所示: 主要写入时要指定每个数据的宽度,若不指定,当修改某行的

  • 本文向大家介绍在 Linux 中查找文件的方法,包括了在 Linux 中查找文件的方法的使用技巧和注意事项,需要的朋友参考一下 使用简单的命令在 Linux 下基于类型、内容等快速查找文件。 如果你是 Windows 或 OSX 的非资深用户,那么可能使用 GUI 来查找文件。你也可能发现界面受限,令人沮丧,或者两者兼而有之,并学会了组织文件并记住它们的确切顺序。你也可以在 Linux 中做到这一

  • 这个问题比原来的标题(“原型和子例程的正向声明”!)简单得多让我们开始吧。我希望答案,无论多么简单,都能帮助我理解子程序/函数、原型和范围以及模块。 在Perl中,子例程几乎可以出现在任何地方,通常不需要进行前向声明(除非子例程声明了一个原型,我不知道如何在Perl中以“标准”的方式执行)。对于我通常使用Perl所做的事情,这些不同的运行的方法几乎没有区别: 我经常使用生成代码,并将其抄录/破解成