CFileDialog 扩展名注意事项
2025-07-01 20:55:10
在 MFC 中,打开和保存对话框都是用 CFileDialog 来实现的,构造函数的第一个参数决定了是"打开"还是"保存"。
#打开对话框
一个简单的对话框如下
1 | CFileDialog dlg(TRUE, nullptr, nullptr, 0, _T("ALL|*.*||")); |
这个对话框有两个问题:
- 用户如果未输入扩展名,
ext将为空。 - 允许用户输入不存在的文件名。
#解决问题一
可以将第二个参数为默认的扩展名,当用户没有输入扩展名的时候,会自动附加默认扩展名。
1 | CFileDialog dlg(TRUE, _T("txt"), nullptr, 0, _T("ALL|*.*||")); |
输入123,会得到文件名123.txt。
- 过滤器优先于默认扩展名。如果用户未输入文件名,但是选择了有效的过滤器:
1 | CFileDialog dlg(TRUE, _T("ini"), nullptr, 0, _T("ALL|*.*|TXT|*.txt||")); |
如果用户选择了txt过滤器,但是输入123作为文件名,将会得到123.txt,而不是123.ini。
- 如果过滤器有多个扩展名,则采用第一个:
1 | CFileDialog dlg(TRUE, _T("ini"), nullptr, 0, _T("ALL|*.*|TXT|*.txt;*.doc||")); |
输入了123作为文件名,得到123.txt。
- 当默认扩展名为
nullptr时,即使选择了过滤器也不会追加扩展名。
1 | CFileDialog dlg(TRUE, nullptr, nullptr, 0, _T("ALL|*.*|TXT|*.txt||")); |
选择过滤器txt,输入123作为文件名将会得到123文件名,而不是123.txt。
#解决问题二
可以在第四个参数上指定OFN_FILEMUSTEXIST,意味着必须输入一个存在的文件名
1 | CFileDialog dlg(TRUE, nullptr, nullptr, OFN_FILEMUSTEXIST, _T("ALL|*.*||")); |
如果用户输入123且文件不存在就会出现警告,这样可以保证DoModal返回时文件名必然有效。

#保存对话框
- 当默认扩展名为
nullptr时,即使选择了过滤器也不会追加扩展名。
1 | CFileDialog dlg(FALSE, nullptr, nullptr, OFN_OVERWRITEPROMPT, _T("TXT|*.txt||")); |
输入123得到123。
- 以用户输入的扩展名为准。
1 | CFileDialog dlg(FALSE, _T(""), nullptr, OFN_OVERWRITEPROMPT, _T("TXT|*.txt||")); |
输入123.doc得到123.doc,而不是123.txt。
#总结
- 如果用户输入了扩展名,则以用户输入为最终值。
- 如果用户未输入扩展名,且设置了默认扩展名,就先采用过滤器中的扩展名,只有用户选择通配符
*.*时才采用默认扩展名。 - 默认扩展名为
nullptr时,在任何情况下都不会自动追加扩展名。
#对于打开对话框
相信没有人会愿意打开不存在的文件,那么OFN_FILEMUSTEXIST几乎是必选项,而这正好也确保了文件名是正确的,所以默认扩展名可有可无。
#对于保存对话框
建议总是设置一个默认扩展名。如果默认扩展名为nullptr,即使选择了有效的过滤器,对话框则也不会追加扩展名,用户必须手动输入才行。