来源:https://www.cnblogs.com/fnlingnzb-learner/p/6424563.html
那么到底如何查找文件呢?我们需要一个结构体和几个大家可能不太熟悉的函数。这些函数和结构体在<io.h>的头文件中,
结构体为struct _finddata_t ,函数为_findfirst、_findnext和_fineclose。
_finddata_t 结构体:存储文件各种信息
unsigned atrrib:文件属性的存储位置。它存储一个unsigned单元,用于表示文件的属性。文件属性是用位表示的,主要有以下一些用无符号整数(2的几次幂)定义的宏:
_A_ARCH(存档)、_A_HIDDEN(隐藏)、_A_NORMAL(正常)、_A_RDONLY(只读)、_A_SUBDIR(文件夹)、_A_SYSTEM(系统)。
这些都是在<io.h>中定义的宏,可以直接使用,而本身的意义其实是一个无符号整型(只不过这个整型应该是2的几次幂,从而保证只有一位为1,而其他位为0)。既然是位表示,那么当一个文件有多个属性时,它往往是通过位或的方式,来得到几个属性的综合。例如只读+隐藏+系统属性,应该为:_A_HIDDEN | _A_RDONLY | _A_SYSTEM 。
time_t time_create:这里的time_t是一个变量类型(长整型?相当于long int?),用来存储时间的,我们暂时不用理它,只要知道,这个time_create变量是用来存储文件创建时间的就可以了。
time_t time_access:文件最后一次被访问的时间。
time_t time_write:文件最后一次被修改的时间。
_fsize_t size:文件的大小。这里的_fsize_t应该可以相当于unsigned整型,表示文件的字节数。
char name[_MAX_FNAME]:文件的文件名。这里的_MAX_FNAME是一个常量宏,它在<stdlib.h>头文件中被定义,表示的是文件名的最大长度。
以此,我们可以推测出,struct _finddata_t ,大概的定义如下:
struct _finddata_t
{
unsigned attrib;
time_t time_create; // -1 for FAT file systems
time_t time_access;
time_t time_write;
_fsize_t size;
char name[_MAX_FNAME];
};
函数:
- long _findfirst( char *filespec, struct _finddata_t *fileinfo );
返回值:如果查找成功的话,将返回一个long型的唯一的查找用的句柄(就是一个唯一编号)。这个句柄将在_findnext函数中被使用。若失败,则返回-1。
filespec:标明文件的字符串,可支持通配符。比如:*.c,则表示当前文件夹下的所有后缀为C的文件。
fileinfo :这里就是用来存放文件信息的结构体的指针。这个结构体必须在调用此函数前声明,不过不用初始化,只要分配了内存空间就可以了。函数成功后,函数会把找到的文件的信息放入这个结构体中。 - int _findnext( long handle, struct _finddata_t *fileinfo );
返回值:若成功返回0,否则返回-1。
handle:即由_findfirst函数返回回来的句柄。
fileinfo:文件信息结构体的指针。找到文件后,函数将该文件信息放入此结构体中。 - int _findclose( long handle );
返回值:成功返回0,失败返回-1。
handle :_findfirst函数返回回来的句柄。
函数实现:
获取某一文件夹下的所有文件名
void getFiles(string path, vector<string>& files)
{
//文件句柄:文件的唯一标识
//long hFile = 0;//32位能跑动
intptr_t hFile = 0;//因为_findfirst和_findnext返回intptr_t而非long型
//定义存储文件信息的结构体
struct _finddata_t fileinfo;
string p;
if ((hFile = _findfirst(p.assign(path).append("\\*").c_str(), &fileinfo)) != -1)//string转化为c_str即C类型的字符串
{
do
{
//如果是目录,迭代之
//如果不是,加入列表
if ((fileinfo.attrib & _A_SUBDIR))//&:attrib和_A_SUBDIR做一次与运算,返回1则为_A_SUBDIR(文件夹)
{
if (strcmp(fileinfo.name, ".") != 0 && strcmp(fileinfo.name, "..") != 0)
getFiles(p.assign(path).append("\\").append(fileinfo.name), files);//递归
}
else
{
files.push_back(p.assign(path).append("\\").append(fileinfo.name));
}
}
while (_findnext(hFile, &fileinfo) == 0);
_findclose(hFile);
}
}
只取文件名,去除后缀(应用于利用数字命名并需要对齐排序的文件)
string removeSuffix(string fileName)
{
const char* full_name = fileName.c_str();
const char* mn_first = full_name;
int tmp = strlen(full_name);//长度不包含'/0'
const char* mn_last = full_name /*+ strlen(full_name)*/;
if (strrchr(full_name, '\\') != NULL)//去除路径,只留文件名
mn_first = strrchr(full_name, '\\') + 1;
else if (strrchr(full_name, '/') != NULL)//去除linux下的文件路径
mn_first = strrchr(full_name, '/') + 1;
if (strrchr(full_name, '.') != NULL)
mn_last = strrchr(full_name, '.');//获取.*后缀
if (mn_last < mn_first)//地址比较
mn_last = full_name + strlen(full_name);
fileName.assign(mn_first, mn_last);//参数10001.jpg和.jpg
return fileName;
}
注释:
函数名称: strrchr
函数原型:char *strrchr(const char *str, char c);
所属库: string.h
函数功能:查找一个字符c在另一个字符串str中末次出现的位置(也就是从str的右侧开始查找字符c首次出现的位置),并返回这个位置的地址。如果未能找到指定字符,那么函数将返回NULL。使用这个地址返回从最后一个字符c到str末尾的字符串。
对应于C++的find_last_of函数,不同点,返回位置索引。