他的代码大致如下:
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
INT_PTR nResult = dlg.DoModal();
我测试了一下,选择比较多txt文件时,确实如他所言,会出现返回值是IDCANCEL的情况,但是有时如果少几个文件,就会返回IDOK。这说明多选文件对话框所选择的文件有一个临界值。选择文件的多少里面体现的应该是一个字符串缓冲区。因此我猜想CFileDialog里面应该有一个字符串缓冲区用于存贮用户所选的文件名,这个缓冲区有一个默认长度,假如所选的全部文件长度超出了默认长度,DoModal函数的返回值是IDCANCEL。如果是这样,那么就有以下一些问题:
1.如果存在这个缓冲区,CFileDialog类中有哪些数据成员负责控制这个缓冲区,这个缓冲区的默认长度又是多少?
2.如何增加这个缓冲区的长度以增加用户选择更多文件的需要?
为此我搜索了一些资料。设置CFileDialog类的初始化值主要集中在m_ofn这个数据成员。
m_ofn
The Windows OPENFILENAME structure. Provides access to basic file dialog box parameters.
其中m_ofn有两个成员负责文件名缓冲区:lpstrFile和nMaxFile。
lpstrFile
指向包含初始化文件名编辑控件使用的文件名的缓冲。如果不需要初始值,这个缓冲的第一个字符必须是NULL。当GetOpenFileName或GetSaveFileName函数返回成功时,这个缓冲包含驱动器,路径,文件名,及所选择的文件的扩展名。
如果OFN_ALLOWMULTISELECT标记被设置并且用户选择了多个文件,缓冲包含了当前目录下被选择文件的文件名。对于Explorer 风格对话框,目录和文件名字符串是被NULL分开的,在文件名之后有一个额外的NULL。对于旧风格对话框,字符串是被空格分开的并且函数为带有空格的文件名使用短文件名。你可以使用FindFirstFile函数在长短文件名之间转换。如果用户只选择了一个文件,lpstrFile字符串在路径和文件名之间没有分隔。
如果缓冲太小,函数返回FALSE并且CommDlgExtendedError函数返回FNERR_BUFFERTOOSMALL.。既然这样,lpstrFile缓冲的首先两个字节包含必需的大小(字节或字符)。
nMaxFile
指定lpstrFile缓冲的大小,以TCHARs为单位。对于ANSI版本,是字节的个数;对于 Unicode版本,是字符的个数。这个缓冲必须足够存储路径和文件名字符串,包含结尾的null字符。如果缓冲太小,GetOpenFileName和GetSaveFileName函数返回假(FALSE)缓冲最小应该在256个字符长。
经过调试观察,我发现nMaxFile的初始值为260。但是我写程序测试这个缓冲区的默认大小时,却和这个初始值有矛盾。
我的测试办法是这样的。首先在E盘建一个Txt Data的文件夹,然后创建40个空的txt文件。创建代码如下:
for (int i = 0;i<40;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
CreateFile(strName, // file to open
GENERIC_READ, // open for reading
FILE_SHARE_READ, // share for reading
NULL, // default security
CREATE_NEW, // existing file only
FILE_ATTRIBUTE_NORMAL, // normal file
NULL); // no attr. template
}
然后我经过多次尝试,发现在选择0..txt,1.txt,2.xtxt,27.txt(共28个文件)时DoModal函数的返回值是IDOK,但是在选择0..txt,1.txt,2.xtxt,27.txt,28.txt(共29个文件)时DoModal函数的返回值是IDCANCEL。接着我计算了一下所选中的文件的总长度(在unicode字符集下编译):
CString strAllFiles = _T(&&);
for (int i = 0;i<28;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是494,如果增加一个28.txt,即:
CString strAllFiles = _T(&&);
for (int i = 0;i<29;i++)
{
CString strName = _T(&&);
strName.Format(_T(&E:Txt Data%d.txt&),i);
strAllFiles = strAllFiles + strName;
}
int nStrLen = strAllFiles.GetLength();
nStrLen的返回值是512.在多字节字符集下也是这个数值。这里需要注意的是CString::GetLength() 对于ASCII,返回字符串所占字节的数目,但如果是Unicode则实际上返回的是字符数而不是字节数
那么我初步断定那个缓冲区的默认大小不是我调试观察到的260,而是512。至于开头如何解决那个问题,只需要定义一个更大的缓冲区,将lpstrFile指向这个缓冲区,重设nMaxFile的值即可,具体是:
TCHAR szLargeBuf[4096]; // 定义一个临时缓冲区
memset(szLargeBuf,'0',4096);
static TCHAR BASED_CODE szFilter[] = _T(&TXT(*.txt)|*.txt||&);
CFileDialog dlg(TRUE,_T(&txt&), NULL,OFN_HIDEREADONLY|OFN_ALLOWMULTISELECT,
szFilter, NULL );
dlg.m_ofn.lpstrFile = szLargeBuf;
#ifdef UNICODE
dlg.m_ofn.nMaxFile = 4096;
#else
dlg.m_ofn.nMaxFile = sizeof (szLargeBuf);
#endif
想读入多文件,但总是最多读入8个文件,超过8个读不进来,设断点检查发现,如果想读入8个文件,程序运行到 if (dlgOpen->DoModal()==IDOK)就跳出if语句,不执行下面的语句。难怪!但是究竟怎么才能读入多个文件那,我搜索DoModal函数源代码,在文件DLGFILE.CPP中找到。发现函数中有个判断语句 DWORD nOffset = lstrlen(m_ofn.lpstrFile)+1; ASSERT(nOffset <= m_ofn.nMaxFile);而nMaxFile最大文件数在构造函数中为空,如果不指定nMaxFile的值,判断语句必然从DoModal函数中跳出。所以我在if (dlgOpen->DoModal()==IDOK)前面写入下面语句CString str; dlgOpen->m_ofn.lpstrFile=str.GetBuffer(100000); str.ReleaseBuffer(); dlgOpen->m_ofn.nMaxFile = 5000;一切搞定! 但是要记住,m_ofn是不可见的,但是写上去不会报错。
CFileDialog如何实现文件多选
CFileDialog类设置OFN_ALLOWMULTISELECT 标志可以实现文件多选功能,但是文件的数量是有限制的,如果要突破这个限制,就必须自己提供缓冲区。例子如下:
CString fileExtensions = "jpg文件(*.jpg) |*.jpg||";
CFileDialog fileDlg(TRUE,
NULL,
NULL,
OFN_ALLOWMULTISELECT | OFN_ENABLESIZING | OFN_HIDEREADONLY,
fileExtensions);
const int MIN_FILE_NUMBER = 10; //至少允许选择10个文件
fileDlg.m_ofn.lpstrFile = new TCHAR[_MAX_PATH * MIN_FILE_NUMBER]; //重新定义缓冲区大小
memset(fileDlg.m_ofn.lpstrFile, 0, _MAX_PATH * MIN_FILE_NUMBER); //初始化定义的缓冲区
fileDlg.m_ofn.nMaxFile = _MAX_PATH * MIN_FILE_NUMBER;
if (IDOK == fileDlg.DoModal())
{
POSITION pos = fileDlg.GetStartPosition();
while (NULL != pos)
{
TRACE(fileDlg.GetNextPathName(pos)); //获取文件名
//使用文件...
}
}
delete[] fileDlg.m_ofn.lpstrFile; //最后别忘了释放内存
CFileDialog设置多选的问题解决
2020-11-17 13:28C语言教程网 C/C++
前几天同事问我在CFileDialog中多选时按确定按钮后DoModal函数的返回值是IDCANCEL的问题解决
延伸 · 阅读
- 2022-03-07小程序实现自定义多层级单选和多选
- 2022-03-03Android ExpandableListView单选以及多选实现代码
- 2022-02-17基于ElementUI中Table嵌套实现多选的示例代码
- 2021-12-05java实现多选批量删除功能
- 2021-03-25利用switch语句进行多选一判断的实例代码
- 2021-03-23iOS tableView实现单选和多选的实例代码
- C/C++
Opencv绘制最小外接矩形、最小外接圆
这篇文章主要为大家详细介绍了Opencv绘制最小外接矩形、最小外接圆的方法,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以...
- C/C++
C语言实现随机抽奖程序
这篇文章主要为大家详细介绍了C语言实现随机抽奖程序,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...
- C/C++
深入分析C++中deque的使用
本篇文章介绍了,深入分析C++中deque的使用。需要的朋友参考下...
- C/C++
如何用C语言、Python实现栈及典型应用
本文先通过实例分别介绍了如何用C语言、Python实现栈,后又介绍栈的典型应用,对大家学习栈很有借鉴参考价值,下面一起来看看吧。...
- C/C++
Visual Studio C++指针靠前靠后的问题全面解析
这篇文章主要介绍了Visual Studio C++指针靠前靠后的问题全面解析,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可...
- C/C++
浅谈防不胜防的unsigned int的运算
下面小编就为大家带来一篇浅谈防不胜防的unsigned int的运算。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...
- C/C++
构造函数不能声明为虚函数的原因及分析
构造函数不需要是虚函数,也不允许是虚函数,因为创建一个对象时我们总是要明确指定对象的类型,尽管我们可能通过实验室的基类的指针或引用去访问...
- C/C++
C++输入一个字符串,把其中的字符按照逆序输出的两种方法解析
以下是对C++中输入一个字符串,把其中的字符按照逆序输出的两种方法进行了详细的分析介绍,需要的朋友可以过来参考下...