为什么递归寻找文件名的程序在64位环境中会崩溃,32位却没有问题,有没有解决方法?

时间:2023-01-11 16:00:29
程序如下

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

void getFiles( string path, string exd, vector<string>& files )
{
//文件句柄
long   hFile   =   0;
//文件信息
struct _finddata_t fileinfo;
string pathName, exdName;

if (0 != strcmp(exd.c_str(), ""))
{
exdName = "\\*." + exd;
}
else
{
exdName = "\\*";
}

if((hFile = _findfirst(pathName.assign(path).append(exdName).c_str(),&fileinfo)) !=  -1)
{
do
{
//如果是文件夹中仍有文件夹,迭代之
//如果不是,加入列表
if((fileinfo.attrib &  _A_SUBDIR))
{
if(strcmp(fileinfo.name,".") != 0  &&  strcmp(fileinfo.name,"..") != 0)
getFiles( pathName.assign(path).append("\\").append(fileinfo.name), exd, files );
}
else
{
if(strcmp(fileinfo.name,".") != 0  &&  strcmp(fileinfo.name,"..") != 0)
files.push_back(pathName.assign(path).append("\\").append(fileinfo.name));
}
}while(_findnext(hFile, &fileinfo)  == 0);
_findclose(hFile);
}
}

int main()
{
string filePath = "D:\\";
vector<string> files;

//获取该路径下的所有jpg文件
getFiles(filePath, "jpg", files);

int size = files.size();
for (int i = 0;i < size;i++)
{
cout<<files[i].c_str()<<endl;
}
return 0;
}

6 个解决方案

#1


先看出错位置,是不是有什么路径拼接处理有问题

#2


system("dir /b /a-d c:\\*.* >d:\\allfiles.txt");
//读文件d:\\allfiles.txt的内容即C:\\下所有文件的名字
system("dir /b /a-d /s c:\\*.* >d:\\allfilesinsub.txt");
//读文件d:\\allfilesinsub.txt的内容即C:\\下所有文件的名字包含子目录
system("dir /b /ad  c:\\*.* >d:\\alldirs.txt");
//读文件d:\\alldirs.txt的内容即C:\\下所有子目录的名字
请记住,能用shell命令获取文件、文件夹信息或者操作文件、文件夹最好用shell命令获取或者操作,而不要用各种API获取或者操作,因为当遇到非法文件夹名或非法文件名或非法文件长度、非法文件日期、压缩文件、链接文件、稀疏文件……等各种意料之外的情况时,API会处理的不全面或陷入死循环,而shell命令不会。
如果嫌system黑窗口一闪,将system("...")替换为WinExec("cmd /c ...",SW_HIDE);

#3


崩溃,
建立crashdump 
或者输出log 查看哪个步骤崩的。


getFiles 跟我写的读目录类似,传vector引用

#4


intptr_t 不能用 long 替换的,还有你写的太啰嗦了。如果是 . 或 .. 必然是带文件夹属性的,这种情况没必要写两次。还有拼接查找到的文件(或文件夹)一次就够了:
#include <string>
#include <io.h>
#include <vector>
#include <iostream>
using namespace std;
 
void getFiles( string path, string ext, vector<string> &files )
{
string find(path);
if (ext.size() > 0)
{
find += "\\*." + ext;
}
else
{
find += "\\*";
}
struct _finddata_t fileinfo;
intptr_t hf = _findfirst(find.c_str(),&fileinfo);
if (hf !=  -1)
{
do {
if ((fileinfo.name == ".") || (fileinfo.name == ".."))
{
continue;
}
string file(path);
file += "\\" + fileinfo.name;
if((fileinfo.attrib &  _A_SUBDIR))
{
getFiles(file, ext, files);
}
else
{
files.push_back(file);
}
} while (_findnext(hf, &fileinfo) == 0);
_findclose(hf);
}
}
 
int main()
{
string filePath = "D:\\";
vector<string> files;
 
//获取该路径下的所有jpg文件
getFiles(filePath, "jpg", files);
 
int size = files.size();
for (int i = 0;i < size;i++)
{
cout<<files[i].c_str()<<endl;
}
return 0;
}

#5


有可能是递归导致的内存溢出。

#6


system("dir /b /a-d c:\\*.* >d:\\allfiles.txt");
//读文件d:\\allfiles.txt的内容即C:\\下所有文件的名字
system("dir /b /a-d /s c:\\*.* >d:\\allfilesinsub.txt");
//读文件d:\\allfilesinsub.txt的内容即C:\\下所有文件的名字包含子目录
system("dir /b /ad  c:\\*.* >d:\\alldirs.txt");
//读文件d:\\alldirs.txt的内容即C:\\下所有子目录的名字
请记住,能用shell命令获取文件、文件夹信息或者操作文件、文件夹最好用shell命令获取或者操作,而不要用各种API获取或者操作,因为当遇到非法文件夹名或非法文件名或非法文件长度、非法文件日期、压缩文件、链接文件、稀疏文件……等各种意料之外的情况时,API会处理的不全面或陷入死循环,而shell命令不会。
如果嫌system黑窗口一闪,将system("...")替换为WinExec("cmd /c ...",SW_HIDE);

#1


先看出错位置,是不是有什么路径拼接处理有问题

#2


system("dir /b /a-d c:\\*.* >d:\\allfiles.txt");
//读文件d:\\allfiles.txt的内容即C:\\下所有文件的名字
system("dir /b /a-d /s c:\\*.* >d:\\allfilesinsub.txt");
//读文件d:\\allfilesinsub.txt的内容即C:\\下所有文件的名字包含子目录
system("dir /b /ad  c:\\*.* >d:\\alldirs.txt");
//读文件d:\\alldirs.txt的内容即C:\\下所有子目录的名字
请记住,能用shell命令获取文件、文件夹信息或者操作文件、文件夹最好用shell命令获取或者操作,而不要用各种API获取或者操作,因为当遇到非法文件夹名或非法文件名或非法文件长度、非法文件日期、压缩文件、链接文件、稀疏文件……等各种意料之外的情况时,API会处理的不全面或陷入死循环,而shell命令不会。
如果嫌system黑窗口一闪,将system("...")替换为WinExec("cmd /c ...",SW_HIDE);

#3


崩溃,
建立crashdump 
或者输出log 查看哪个步骤崩的。


getFiles 跟我写的读目录类似,传vector引用

#4


intptr_t 不能用 long 替换的,还有你写的太啰嗦了。如果是 . 或 .. 必然是带文件夹属性的,这种情况没必要写两次。还有拼接查找到的文件(或文件夹)一次就够了:
#include <string>
#include <io.h>
#include <vector>
#include <iostream>
using namespace std;
 
void getFiles( string path, string ext, vector<string> &files )
{
string find(path);
if (ext.size() > 0)
{
find += "\\*." + ext;
}
else
{
find += "\\*";
}
struct _finddata_t fileinfo;
intptr_t hf = _findfirst(find.c_str(),&fileinfo);
if (hf !=  -1)
{
do {
if ((fileinfo.name == ".") || (fileinfo.name == ".."))
{
continue;
}
string file(path);
file += "\\" + fileinfo.name;
if((fileinfo.attrib &  _A_SUBDIR))
{
getFiles(file, ext, files);
}
else
{
files.push_back(file);
}
} while (_findnext(hf, &fileinfo) == 0);
_findclose(hf);
}
}
 
int main()
{
string filePath = "D:\\";
vector<string> files;
 
//获取该路径下的所有jpg文件
getFiles(filePath, "jpg", files);
 
int size = files.size();
for (int i = 0;i < size;i++)
{
cout<<files[i].c_str()<<endl;
}
return 0;
}

#5


有可能是递归导致的内存溢出。

#6


system("dir /b /a-d c:\\*.* >d:\\allfiles.txt");
//读文件d:\\allfiles.txt的内容即C:\\下所有文件的名字
system("dir /b /a-d /s c:\\*.* >d:\\allfilesinsub.txt");
//读文件d:\\allfilesinsub.txt的内容即C:\\下所有文件的名字包含子目录
system("dir /b /ad  c:\\*.* >d:\\alldirs.txt");
//读文件d:\\alldirs.txt的内容即C:\\下所有子目录的名字
请记住,能用shell命令获取文件、文件夹信息或者操作文件、文件夹最好用shell命令获取或者操作,而不要用各种API获取或者操作,因为当遇到非法文件夹名或非法文件名或非法文件长度、非法文件日期、压缩文件、链接文件、稀疏文件……等各种意料之外的情况时,API会处理的不全面或陷入死循环,而shell命令不会。
如果嫌system黑窗口一闪,将system("...")替换为WinExec("cmd /c ...",SW_HIDE);