(19):浏览和打开文件
最后更新于:2022-04-01 20:07:47
在应用程序中,我们很经常要实现的功能,是Open文件或保存文件对话框,让用户来选择一个或N个文件。本文我将介绍两种思路,第一种方法较为复杂,第二种方法较为简单。
### 方法一:老规矩
这是一种传统方法,使用GetOpenFileName或者GetSaveFileName函数,看名字就知道,前者用来打开文件,后者是保存文件,这两个函数的用法是一样的,因此,我只演示其中一个。
无论你使用哪个函数,都要涉及一个结构体——OPENFILENAME。关于它的成员,我就不一一来说了,挑几个有用的扯一扯。
lStructSize:结构的大小,弄个sizeof就行了。
lpstrFilter:设置过滤器。注意啊,这个过滤器字符串和.NET中的写法不同,.NET的写法是从VB6继承过来,可以写成“垃圾文件 | *.rbs | 老鼠文件 | *.mos”,我们这里不是用“|”来分隔的,而是用“\0”分隔,而**结尾是两个NULL,即两个“\0”**。
nFilterIndex:过滤器索引,设置了N个滤过的后缀,设置默认选择哪个,第一个为1,第二个为2,第三个为3,依此类推,是从1开始的,不是0。
lpstrFile:文件名,包含完整路径,当对话框关闭后,我们就是从这个成员把用户选择的文件名取出。
nMaxFile:文件名的长度,一定要足够大,不然就装不下了,对路径长度,系统是有限定的,用MAX_PATH宏就可以了,WCHAR myFilename[MAX_PATH]。
Flags:标志位。主要决定对话框应具备哪些特征和行为,如是否检查目标文件已存在。
哦,这个可能看了没感觉,看看例子代码,一目了然。
~~~
OPENFILENAME opfn;
WCHAR strFilename[MAX_PATH];//存放文件名
//初始化
ZeroMemory(&opfn, sizeof(OPENFILENAME));
opfn.lStructSize = sizeof(OPENFILENAME);//结构体大小
//设置过滤
opfn.lpstrFilter = L"所有文件\0*.*\0文本文件\0*.txt\0MP3文件\0*.mp3\0";
//默认过滤器索引设为1
opfn.nFilterIndex = 1;
//文件名的字段必须先把第一个字符设为 \0
opfn.lpstrFile = strFilename;
opfn.lpstrFile[0] = '\0';
opfn.nMaxFile = sizeof(strFilename);
//设置标志位,检查目录或文件是否存在
opfn.Flags = OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST;
//opfn.lpstrInitialDir = NULL;
// 显示对话框让用户选择文件
if(GetOpenFileName(&opfn))
{
//在文本框中显示文件路径
HWND hEdt = GetDlgItem(hDlg, IDC_EDTFILENAME);
SendMessage(hEdt, WM_SETTEXT, NULL, (LPARAM)strFilename);
}
~~~
ZeroMemory函数前面说过了,你就当作它用来初始化结构就行了。
你可能会疑问,不是说lpstrFilter的字符串是两个NULL结尾的吗,为什么代码中只有一个?因为你输入的字符串会在后面自动加了个‘\0’,所以我们加一个就OK,后面自动加上一个,就两个了。
下图是执行结果:
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd3116f8b1.PNG)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd3118f933.PNG)
GetSaveFileName的用法也一样,有兴趣的朋友可以回家自己玩一下。
### 方法二:新规矩
这个新规矩是谁定的?哈哈,我定的。
都说知识是可以综合运用的,所以,下面的方法是灵活运用知识了。为啥?记得吧,.NET类库其实也为我们封装了,大家玩过Windows Forms开发肯定知道,在System.Windows.Forms下面的。是啊,既然有了现成的,我们干吗不拿来用呢。
首先,打开项目属性,按下图的方法,让项目支持 /clr ,然后点击“应用”。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd311a1708.PNG)
引用需要用到的程序集。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd311b96c2.PNG)
在代码文件中引入命名空间。
~~~
using namespace System;
using namespace System::Windows::Forms;
~~~
然后实现打开文件的功能。
~~~
if(LOWORD(wParam) == IDC_BTNOPWTCLR)
{
OpenFileDialog ^dlg = gcnew OpenFileDialog;
dlg ->Filter = L"所有文件|*.*|图片文件|*.jpg";
dlg->FilterIndex = 1;
dlg->CheckFileExists = true;
if (dlg ->ShowDialog() == DialogResult::OK)
{
//取得文件名
}
}
~~~
但是,当获取文件名的时候,遇到了一个严重问题了,在CLR中,我们取到的文件名是System::String ^类型的,可是我们这里的Win32程序需要wchar_t类型的字符串,这怎么办?能进行转换吗?
微软早就想到这个问题,所以,它给我们写了一个vcclr.h头文件,里面有一个PtrToStringChars函数,使用它就可以将托管的字符串转为标准字符串了。
~~~
if (dlg ->ShowDialog() == DialogResult::OK)
{
//取得文件名
pin_ptr strFileName = PtrToStringChars(dlg->FileName);
HWND hEdt = GetDlgItem(hDlg, IDC_EDTFILENAME);
SendMessage(hEdt, WM_SETTEXT, NULL, (LPARAM)strFileName);
}
~~~
但是,问题仍未解决,运行之后你会很惊喜地收到一个异常信息。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd311cc054.PNG)
运气真的太好了,那么这个怎么解决呢?再次打开项目属性窗口,找到“链接器”-“高级”,把CLR线程模型改为STA就可以了。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd311dfe4e.PNG)
现在运行应用程序,估计没有问题了。
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd31278e4b.PNG)
![](https://docs.gechiui.com/gc-content/uploads/sites/kancloud/2016-06-14_575fd31297003.PNG)
代码我随后上传。
好了,好了,现在大功告成了。拜拜。
';