他的代碼大致如下:
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; //最後別忘了釋放內存