利用C++获取某一文件夹下的所有文件名,并获取不带后缀的文件名字

来源: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];
};

函数:

  1. long _findfirst( char *filespec, struct _finddata_t *fileinfo ); 
    返回值:如果查找成功的话,将返回一个long型的唯一的查找用的句柄(就是一个唯一编号)。这个句柄将在_findnext函数中被使用。若失败,则返回-1。
    filespec:标明文件的字符串,可支持通配符。比如:*.c,则表示当前文件夹下的所有后缀为C的文件。
    fileinfo :这里就是用来存放文件信息的结构体的指针。这个结构体必须在调用此函数前声明,不过不用初始化,只要分配了内存空间就可以了。函数成功后,函数会把找到的文件的信息放入这个结构体中。
  2. int _findnext( long handle, struct _finddata_t *fileinfo );
    返回值:若成功返回0,否则返回-1。
     handle:即由_findfirst函数返回回来的句柄。
    fileinfo:文件信息结构体的指针。找到文件后,函数将该文件信息放入此结构体中。
  3. 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函数,不同点,返回位置索引。

猜你喜欢

转载自blog.csdn.net/vict_wang/article/details/83997256