功能:对指定文件执行某个操作
原型:
BOOL ShellExecuteEx(LPSHELLEXECUTEINFO lpExecInfo);
也可以使用ShellExecute,比如下面为使用ShellExecute打开指定文件(使用默认程序):
ShellExecute(nullptr, _T("open"), _T("C:\\abc.wav"), _T(""), _T(""), SW_SHOW);
参数:
lpExecInfo
一个指向SHELLEXECUTEINFO结构的指针,用来传递和保存应用程序执行相关的信息。
返回值:
如果函数成功执行就返回TRUE,否则返回 FALSE。可调用GetLastError获取错误信息。
备注:
由于ShellExecuteEx能够将执行委托给那些由组件对象模型COM激活的Shell扩展(数据源,上下文菜单句柄,动词实现),因此在调用ShellExecuteEx之前要先初始化 COM。某些Shell扩展要求单线程单元模型的COM,在这种情况下,应当像下面一般初始化COM:
CoInitializeEx (NULL,COINIT_APARTMENTTHREADED | COINIT_DISABLE_OLE1DDE)
在某些情况下 ShellExecuteEx 并没有使用这种类型的Shell 扩展,这时就无需初始化COM。虽然如此,总是在使用这个函数之前初始化COM是个不错的举措。
在MSDN中,它这样定义:
原型如下:
typedef struct _SHELLEXECUTEINFO {
DWORD cbSize;
ULONG fMask;
HWND hwnd;
LPCTSTR lpVerb;
LPCTSTR lpFile;
LPCTSTR lpParameters;
LPCTSTR lpDirectory;
int nShow;
HINSTANCE hInstApp;
LPVOID lpIDList;
LPCTSTR lpClass;
HKEY hkeyClass;
DWORD dwHotKey;
union {
HANDLE hIcon;
HANDLE hMonitor;
} DUMMYUNIONNAME;
HANDLE hProcess;
} SHELLEXECUTEINFO, *LPSHELLEXECUTEINFO;
参数详解:
cbSize
结构大小,以字节为单位。
fMask
一个标志数组,用来设置其他成员的有效性。
hwnd
可选。执行ShellExecuteEx的窗口句柄,可设为NULL。
lpVerb
指定执行的动作,包括:edit ,explore ,find ,open,print, properties,为NULL则默认为open
lpFile
以\0 结尾的字符串,指出 lpVerb 的操作对象的路径,被系统支持的操作包括文本的 open 、 print等
lpParameters
可选。运行/打开程序的参数,如果打开的是一个文档,则该项无效
lpDirectory
可选。指明工作目录的名字,成员没有说明,则默认为当前目录
nShow
必须。指定打开的程序的显示方式,为SW_值中的一个。
hInstApp
【out】如果设置SEE_MASK_NOCLOSEPROCESS S值并且ShellExecuteEx 调用成功,则该项的值大于32,如果调用失败,则将设置为 SE_ERR_XXX 的错误值。
lpIDList
一个ITEMIDLIST结构的地址,用来存储成员的特别标识符,当fMask不包括SEE_MASK_IDLIST
或SEE_MASK_INVOKEIDLIST
时该项被忽略
lpClass
用以指明文件类别的名字或GUID,当fMask不包括SEE_MASK_CLASSNAME
时该项被忽略
hkeyClass
获得已在系统注册的文件类型的Handle,当fMask不包括SEE_MASK_HOTKEY
时该项被忽略
dwHotKey
程序的热键关联,低位存储虚拟关键码(Key Code),高位存储修改标志位(HOTKEYF_),修改标志为(modifier flags)的详细列表请看WM_SETHOTKEY消息的描述,当fmask不包括SEE_MASK_HOTKEY
时该项被忽略
DUMMYUNIONNAME
hIcon
取得对应文件类型的图标的Handle,当fMask不包括SEE_MASK_ICON时该项被忽略
hMonitor
将文档显示在显示器上的Handle,当fMask不包括SEE_MASK_HMONITOR时该项被忽略
hProcess
指向新启动的程序的句柄。若fMask不设为SEE_MASK_NOCLOSEPROCESS
则该项值为NULL。但若新程序没有启动,即使fMask设为SEE_MASK_NOCLOSEPROCESS
,该值也仍为NULL。
例:
1、打开一个应用程序或文档直到其运行结束
SHELLEXECUTEINFO ShellInfo;
memset(&ShellInfo, 0, sizeof(ShellInfo));
ShellInfo.cbSize = sizeof(ShellInfo); //大小
ShellInfo.lpVerb = _T("open"); //运行
ShellInfo.lpFile = szFilePath; // 带路径的执行文件名
ShellInfo.nShow = SW_SHOWNORMAL; //正常显示
ShellInfo.fMask = SEE_MASK_NOCLOSEPROCESS | SEE_MASK_FLAG_NO_UI;//保存新程序句柄到hProcess,不显示错误提示框
BOOL bResult = ShellExecuteEx(&ShellInfo);
WaitForSingleObject(ShellInfo.hProcess,INFINITE);
另外打开另一个程序还有system(如system("shutdown -s -t 10")为打开shutdown进程)、CreateProcess方法。
2、如何打开一个网页
SHELLEXECUTEINFO ShellInfo;
memset(&ShellInfo, 0, sizeof(ShellInfo));
ShellInfo.cbSize = sizeof(ShellInfo);
ShellInfo.lpVerb = _T("open");
ShellInfo.lpFile = _T("http://www.sina.com");
ShellInfo.nShow = SW_SHOWNORMAL;
ShellInfo.fMask = SEE_MASK_NOCLOSEPROCESS | SEE_MASK_FLAG_NO_UI;
BOOL bResult = ShellExecuteEx(&ShellInfo);
原文地址:http://www.cppblog.com/SpringSnow/archive/2009/04/22/80719.html
3、WinExec()、CreateProcess()
打开另一个进程的方法还有WinExec()和CreateProcess(),其中以WinExec最为简单,ShellExecute比WinExec灵活一些,CreateProcess最为复杂。
WinExec() 有两个参数,前一个指定路径(文件名加上可选参数),后一个指定显示方式,如WinExec(“Notepad.exe”, SW_SHOW); // 打开记事本,WinExec(“D://Program Files//Test//Test.exe”,SW_SHOWMAXIMIZED); // 以最大化的方式打Test.exe。
windows推荐使用CreateProcess()替换该方法,而且因为WinExec()没有宽字符版本,所以如果想要传递宽字符的参数,或者要打开进程的WinMain()是宽字符版本(参数lpCmdLine是LPWSTR类型),那么使用WinExec()就会出错。
ShellExecute() 可以指定工作目录,并且还可以寻找文件的关联直接打开不用加载与文件关联的应用程序,ShellExecute还可以打开网页,启动相应的邮件关联发送邮件等等。
CreateProcess() 一共有十个参数,不过大部分都可以用NULL代替,它可以指定进程的安全属性,继承信息,类的优先级等等。
4、关闭指定进程
使用taskkill命令:
String cmdLine;
cmdLine << L"taskkill /f /im " << strProcessName << L" /t";
WinExec(cmdLine.toStdString().c_str(), SW_HIDE);
使用TerminateProcess:
bool killClientProcess(const std::string& processName)
{
bool ret = true;
DWORD pid;
while (findProcess(processName, pid)) {
if (!killProcess(pid))
ret = false;
}
return ret;
}
bool findProcess(const std::string strProcessName, std::vector<DWORD>& vPid)
{
if (strProcessName.empty()) return false;
// 遍历进程
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
if (hSnapshot == INVALID_HANDLE_VALUE)
return false;
PROCESSENTRY32 ps = { 0 };
ps.dwSize = sizeof(PROCESSENTRY32);
if (!Process32First(hSnapshot, &ps))
return false;
do {
if (lstrcmpi(ps.szExeFile, strProcessName.c_str()) == 0) { // 查找进程
vPid.push_back(ps.th32ProcessID);
}
} while (Process32Next(hSnapshot, &ps));
CloseHandle(hSnapshot);
return true;
}
bool killProcess(DWORD dwPid)
{
//关闭进程
HANDLE killHandle = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION | // Required by Alpha
PROCESS_CREATE_THREAD | // For CreateRemoteThread
PROCESS_VM_OPERATION | // For VirtualAllocEx/VirtualFreeEx
PROCESS_VM_WRITE, // For WriteProcessMemory);
FALSE, dwPid);
if (killHandle == NULL)
return false;
TerminateProcess(killHandle, 0);
return true;
}
5、打开文件所在的目录并定位文件
#include "boost/filesystem.hpp"
bool openContainingFolder(const std::wstring& filePath) {
namespace fs = boost::filesystem;
// 检查文件是否存在
if (!fs::exists(filePath)) {
return false;
}
// 构造参数:/select参数需要完整文件路径
std::wstring params = L"/select,\"" + fs::path(filePath).wstring() + L"\"";
// 调用ShellExecute
HINSTANCE result = ShellExecuteW(
NULL, // 无父窗口
L"open", // 执行"打开"操作
L"explorer.exe", // 目标程序
params.c_str(), // 参数
NULL, // 默认工作目录
SW_SHOW // 窗口显示方式
);
// 返回值大于32表示成功
return reinterpret_cast<int>(result) > 32;
}