在編程過程中有個需求,點擊某個按鈕需要彈出系統的聲音控制面板。在網上查了下代碼中調用其他exe程序或者打開其他文件的方法。
自己借鑒網上的文章稍微總結下,加深下印象,也給方便自己用。
在代碼中調用其他exe程序或者打開其他功能末班的方法主要有以下幾種:
1.使用system函數
system函數,函數原型system(執行shell命令)定義函數是,int system(const char * string),可以調用一些DOS命令,比如:
system("cls");//清屏,等於在DOS上使用cls命令
例如:system("pause");//暫停當前的進程
system("dir"); //列出當前項目文件的目錄
system函數會調用fork()函數產生子進程,由子進程來調用/bin/sh-c string來執行參數string字符串所代表的命令,此命令執行完后隨即返回調用的進程,system函數不能控制程序窗口是否顯示。一般在項目中,這種方法不予考慮。
2.使用execl或execv函數
這是exec函數族里面的兩個函數,在UNIX環境下用的比較多,exec函數族的作用是根據指定的文件名找到可執行文件,並用它來取代調用進程的內容,也就是說在調用進程內部執行一個可執行文件,這里的可執行文件既可以是二進制文件,也可以是任何Linux下的可執行的腳本文件,與一般情況不同,exec函數族的函數執行成功后不會返回,只有調用失敗了才會返回一個-1,從原程序的調用點接着往下執行。它也是不能控制程序窗口是否顯示。
3.WinExec函數
WinExec函數可以方便地在應用程序中執行一個程序,執行一個標准程序一般使用:
WinExec('C:\WINDOWS\NOTEPAD.EXE', SW_SHOWNORMAL);
第二個參數是控制程序主窗口的顯示方式,用WinExec只能執行exe文件,但是WinExec函數不好控制主程序來等該exe程序的結束。WinExec是Windows的老函數了,現在的CreateProcess幾乎可以替代WinExe的所有功能。
4.ShellExcecuteEx函數
這個函數主要是在文件上執行一個函數,如果執行成功則返回一個非0值,否則返回一個0值。函數的原型是:
WINSHELLAPI BOOL WINAPI ShellExecuteEx(
LPSHELLEXECUTEINFO lpExecInfo);
參數lpExecInfo:是數據結構 LPSHELLEXECUTEINFO 的地址,包含了和接收一些被執行的應用程序的一些信息。下面是一個使用的示例:
SHELLEXECUTEINFO ShExecInfo = {0};
ShExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
ShExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS;
ShExecInfo.hwnd = NULL;
ShExecInfo.lpVerb = _T("open");
ShExecInfo.lpFile = _T("a.exe");
ShExecInfo.lpParameters = _T("-a a");
ShExecInfo.lpDirectory = NULL;
ShExecInfo.nShow = SW_HIDE;
ShExecInfo.hInstApp = NULL;
ShellExecuteEx(&ShExecInfo);
AfxGetApp()->BeginWaitCursor();
WaitForSingleObject(ShExecInfo.hProcess,INFINITE);
AfxGetApp()->EndWaitCursor();
這個函數可以通過ShExecInfo.lpParameters來傳遞exe文件的參數,通過屬性ShExecInfo.nShow來控制程序窗口是否顯示。
后面三個函數都是SDK函數,現在在執行項目中的exe文件的時候,使用比較多的就是后面得兩個函數,我在自己的項目中使用的就是ShellExecuteEx函數,比CreateProcess函數使用方便一些。
5.使用CreateProcess函數
CreateProcess函數是新建一個進程,下面是一個使用CreateProcess函數的例子:
void createchildprocessandwaituntildone(const ansistring& strcmdline)
{
process_information piprocinfo;
startupinfo sistartinfo;
// set up members of startupinfo structure.
sistartinfo.cb = sizeof(startupinfo);
sistartinfo.lpreserved = null;
sistartinfo.lpreserved2 = null;
sistartinfo.cbreserved2 = 0;
sistartinfo.lpdesktop = null;
sistartinfo.dwflags = 0;
// create the child process.
createprocess(
null,
strcmdline.c_str(),
null, // process security attributes
null, // primary thread security attributes
0, // handles are inherited
0, // creation flags
null, // use parent\'s environment
null, // use parent\'s current directory
&sistartinfo, // startupinfo pointer
&piprocinfo); // receives process_information
// wait for the processs to finish
dword rc = waitforsingleobject(
piprocinfo.hprocess, // process handle
infinite);
}
使用CreateProcess函數,你必須人為地控制該進程的生死,這一方面是使用進程控制來實現。進程控制的意義在於可以創建一個進程,並且可以通過進程句柄結束進程。
VC中調用其他可執行程序,就我目前所知有三種方法:
HINSTANCE ShellExecute(HWND hwnd, LPCTSTRlpVerb,LPCTSTRlpFile,LPCTSTRlpParameters,LPCTSTRlpDirectory,INTnShowCmd);
特殊需要講的參數:
lpVerb:對lpFile要執行的行為,包括"open","explore","print","edit","find","properties"
lpFile:要操作的對象
lpParameters:傳遞給為lpFile的參數
nShowCmd:窗口的顯示參數,不顯示(SW_HIDE)
注意:返回值如果小於32,則是錯誤,見具體錯誤
其擴展函數ShellExecuteEx功能更全
此函數聲明包含在頭文件shellapi.h,庫是shell32.lib- UINT WinExec(
LPCSTR lpCmdLine, // command line
UINT uCmdShow // window style
);
參數lpCmdLine為文件名+參數
如果執行正確,返回值大於31
聲明在頭文件windows.h,包含在kernel32.lib中 - BOOL CreateProcess(
LPCTSTR lpApplicationName, // name of executable module
LPTSTR lpCommandLine, // command line string
LPSECURITY_ATTRIBUTES lpProcessAttributes, // SD
LPSECURITY_ATTRIBUTES lpThreadAttributes, // SD
BOOL bInheritHandles, // handle inheritance option
DWORD dwCreationFlags, // creation flags
LPVOID lpEnvironment, // new environment block
LPCTSTR lpCurrentDirectory, // current directory name
LPSTARTUPINFO lpStartupInfo,// startup information
LPPROCESS_INFORMATION lpProcessInformation // process information
);
這個夠復雜,啟動一個進程的方式打開一個程序。聲明所在的頭文件和庫同WinExec
以上三種方法,推薦使用第一種。第二種不推薦使用,因為它存在只是為了支持16位機的。
還需要考慮另外一個問題,就是后面程序的執行,可能需要等待這個進程結束之后才能執行,而不能兩個進程並發執行。(兩個進程一個是原進程,一個是啟動的進程)。參考了:http://www.cnblogs.com/del/archive/2008/02/13/1067871.html
可以如下使用:
HANDLE hProg;
hProg=ShellExecute(NULL, "open", "xxx.exe", NULL, NULL, SW_HIDE);
WaitForSingleObject(hProg, INIFINTE);
在我使用的時候是要用一個MFC程序打開一個控制台程序,控制台程序執行結果寫入到一個文檔中,然后再用mfc程序讀出。(為什么這么麻煩,是因為控制台程序用到的庫是第三方庫,不支持MFC,沒辦法)。
在控制台程序結束時,寫的文件未必已經寫完,並關閉了。所以還需要額外做些工作。
While(!FileExists(Result.TXT)) ;
READFile(Result.TXT);
注意while語句后面的";"
ShellExecute:
1.函數功能:
你可以給它任何文件的名字,它都能識別出來並打開它。
2.函數原型:
HINSTANCE ShellExecute(
HWND hwnd,
LPCTSTR lpOperation,
LPCTSTR lpFile,
LPCTSTR lpParameters,
LPCTSTR lpDirectory,
INT nShowCmd
);
3.參數說明:
hwnd:
用於指定父窗口句柄。當函數調用過程出現錯誤時,它將作為Windows消息窗口的父窗口。
lpOperation:
用於指定要進行的操作。
“open”操作表示執行由lpFile參數指定的程序,或打開由lpFile參數指定的文件或文件夾;
“print”操作表示打印由lpFile參數指定的文件;
“explore”操作表示瀏覽由lpFile參數指定的文件夾。
當參數設為NULL時,表示執行默認操作“open”。
lpFile:
用於指定要打開的文件名、要執行的程序文件名或要瀏覽的文件夾名。
lpParameters:
若lpFile參數是一個可執行程序,則此參數指定命令行參數,否則此參數應為NULL.
lpDirectory:
用於指定默認目錄.
nShowCmd:
若lpFile參數是一個可執行程序,則此參數指定程序窗口的初始顯示方式,否則此參數應設置為0。
這個參數常用的常數:
SW_HIDE 隱藏窗口,活動狀態給令一個窗口
SW_MINIMIZE 最小化窗口,活動狀態給令一個窗口
SW_RESTORE 用原來的大小和位置顯示一個窗口,同時令其進入活動狀態
SW_SHOW 用當前的大小和位置顯示一個窗口,同時令其進入活動狀態
SW_SHOWMAXIMIZED 最大化窗口,並將其激活
SW_SHOWMINIMIZED 最小化窗口,並將其激活
SW_SHOWMINNOACTIVE 最小化一個窗口,同時不改變活動窗口
SW_SHOWNA 用當前的大小和位置顯示一個窗口,不改變活動窗口
SW_SHOWNOACTIVATE 用最近的大小和位置顯示一個窗口,同時不改變活動窗口
SW_SHOWNORMAL 與SW_RESTORE相同
若ShellExecute函數調用成功,則返回值為被執行程序的實例句柄。若返回值小於32,則表示出現錯誤。
4.使用方法:
例如:
ShellExecute(NULL,"open","iloveu.bmp",NULL,NULL,SW_SHOWNORMAL);
用缺省的位圖編輯器打開一個叫iloveu.bmp的位圖文件,這個缺省的位圖編輯器可能是 Microsoft Paint, Adobe Photoshop, 或者 Corel PhotoPaint。
這個函數能打開任何文件,甚至是桌面和URL快捷方式( .ink或 .url)。ShellExecute解析系統注冊表HKEY_CLASSES_ROOT中所有的內容,判斷啟動那一個執行程序,並且啟動一個新的實例或使用DDE將文件名連到一打開的實例。然后,ShellExecute 返回打開文件的應用的實例句柄。
ShellExecute(NULL, "open", "http://www.microsoft.com", NULL, NULL, SW_SHOWNORMAL);
這個代碼使你能訪問微軟的主頁。當ShellExecute遇到文件名前面的“http:”時,可以判斷出要打開的文件是Web文件,隨之啟動Internet Explorer 或者 Netscape Navigator 或者任何你使用的別的瀏覽器打開文件。
ShellExecute還能識別其它協議,象FTP、GOPHER。甚至識別“mailto”,如果文件名指向“mailto:zxn@hq.cninfo.net”,它啟動電子郵件程序並打開一個待編輯的新郵件,例如:
ShellExecute(NULL, "open",“mailto:zxn@hq.cninfo.net”, NULL, NULL, SW_SHOWNORMAL);打開新郵件窗口。
總之,ShellExecute函數就是如此簡單地打開磁盤文件和Internet文件。如果將第二個參數“OPEN”改為“PRINT”或者“EXPLORE”,ShellExecute將能打印文件和打開文件夾。ShellExecute還有一個擴展函數ShellExecuteEx,所帶參數中有一個特殊的結構,功能更強,或者任何你使用的別的瀏覽器打開文件。
======================================================================================
VC中調用其他可執行程序,就我目前所知有三種方法:
HINSTANCE ShellExecute(HWND hwnd, LPCTSTRlpVerb,LPCTSTRlpFile,LPCTSTRlpParameters,LPCTSTRlpDirectory,INTnShowCmd);
特殊需要講的參數:
lpVerb:對lpFile要執行的行為,包括"open","explore","print","edit","find","properties"
lpFile:要操作的對象
lpParameters:傳遞給為lpFile的參數
nShowCmd:窗口的顯示參數,不顯示(SW_HIDE)
注意:返回值如果小於32,則是錯誤,見具體錯誤
其擴展函數ShellExecuteEx功能更全
此函數聲明包含在頭文件shellapi.h,庫是shell32.lib- UINT WinExec(
LPCSTR lpCmdLine, // command line
UINT uCmdShow // window style
);
參數lpCmdLine為文件名+參數
如果執行正確,返回值大於31
聲明在頭文件windows.h,包含在kernel32.lib中 - BOOL CreateProcess(
LPCTSTR lpApplicationName, // name of executable module
LPTSTR lpCommandLine, // command line string
LPSECURITY_ATTRIBUTES lpProcessAttributes, // SD
LPSECURITY_ATTRIBUTES lpThreadAttributes, // SD
BOOL bInheritHandles, // handle inheritance option
DWORD dwCreationFlags, // creation flags
LPVOID lpEnvironment, // new environment block
LPCTSTR lpCurrentDirectory, // current directory name
LPSTARTUPINFO lpStartupInfo,// startup information
LPPROCESS_INFORMATION lpProcessInformation // process information
);
這個夠復雜,啟動一個進程的方式打開一個程序。聲明所在的頭文件和庫同WinExec
以上三種方法,推薦使用第一種。第二種不推薦使用,因為它存在只是為了支持16位機的。
還需要考慮另外一個問題,就是后面程序的執行,可能需要等待這個進程結束之后才能執行,而不能兩個進程並發執行。(兩個進程一個是原進程,一個是啟動的進程)。參考了:http://www.cnblogs.com/del/archive/2008/02/13/1067871.html
可以如下使用:
HANDLE hProg;
hProg=ShellExecute(NULL, "open", "xxx.exe", NULL, NULL, SW_HIDE);
WaitForSingleObject(hProg, INIFINTE);
在我使用的時候是要用一個MFC程序打開一個控制台程序,控制台程序執行結果寫入到一個文檔中,然后再用mfc程序讀出。(為什么這么麻煩,是因為控制台程序用到的庫是第三方庫,不支持MFC,沒辦法)。
在控制台程序結束時,寫的文件未必已經寫完,並關閉了。所以還需要額外做些工作。
While(!FileExists(Result.TXT)) ;
READFile(Result.TXT);
注意while語句后面的";"
ShellExecute:
1.函數功能:
你可以給它任何文件的名字,它都能識別出來並打開它。
2.函數原型:
HINSTANCE ShellExecute(
HWND hwnd,
LPCTSTR lpOperation,
LPCTSTR lpFile,
LPCTSTR lpParameters,
LPCTSTR lpDirectory,
INT nShowCmd
);
3.參數說明:
hwnd:
用於指定父窗口句柄。當函數調用過程出現錯誤時,它將作為Windows消息窗口的父窗口。
lpOperation:
用於指定要進行的操作。
“open”操作表示執行由lpFile參數指定的程序,或打開由lpFile參數指定的文件或文件夾;
“print”操作表示打印由lpFile參數指定的文件;
“explore”操作表示瀏覽由lpFile參數指定的文件夾。
當參數設為NULL時,表示執行默認操作“open”。
lpFile:
用於指定要打開的文件名、要執行的程序文件名或要瀏覽的文件夾名。
lpParameters:
若lpFile參數是一個可執行程序,則此參數指定命令行參數,否則此參數應為NULL.
lpDirectory:
用於指定默認目錄.
nShowCmd:
若lpFile參數是一個可執行程序,則此參數指定程序窗口的初始顯示方式,否則此參數應設置為0。
這個參數常用的常數:
SW_HIDE 隱藏窗口,活動狀態給令一個窗口
SW_MINIMIZE 最小化窗口,活動狀態給令一個窗口
SW_RESTORE 用原來的大小和位置顯示一個窗口,同時令其進入活動狀態
SW_SHOW 用當前的大小和位置顯示一個窗口,同時令其進入活動狀態
SW_SHOWMAXIMIZED 最大化窗口,並將其激活
SW_SHOWMINIMIZED 最小化窗口,並將其激活
SW_SHOWMINNOACTIVE 最小化一個窗口,同時不改變活動窗口
SW_SHOWNA 用當前的大小和位置顯示一個窗口,不改變活動窗口
SW_SHOWNOACTIVATE 用最近的大小和位置顯示一個窗口,同時不改變活動窗口
SW_SHOWNORMAL 與SW_RESTORE相同
若ShellExecute函數調用成功,則返回值為被執行程序的實例句柄。若返回值小於32,則表示出現錯誤。
4.使用方法:
例如:
ShellExecute(NULL,"open","iloveu.bmp",NULL,NULL,SW_SHOWNORMAL);
用缺省的位圖編輯器打開一個叫iloveu.bmp的位圖文件,這個缺省的位圖編輯器可能是 Microsoft Paint, Adobe Photoshop, 或者 Corel PhotoPaint。
這個函數能打開任何文件,甚至是桌面和URL快捷方式( .ink或 .url)。ShellExecute解析系統注冊表HKEY_CLASSES_ROOT中所有的內容,判斷啟動那一個執行程序,並且啟動一個新的實例或使用DDE將文件名連到一打開的實例。然后,ShellExecute 返回打開文件的應用的實例句柄。
ShellExecute(NULL, "open", "http://www.microsoft.com", NULL, NULL, SW_SHOWNORMAL);
這個代碼使你能訪問微軟的主頁。當ShellExecute遇到文件名前面的“http:”時,可以判斷出要打開的文件是Web文件,隨之啟動Internet Explorer 或者 Netscape Navigator 或者任何你使用的別的瀏覽器打開文件。
ShellExecute還能識別其它協議,象FTP、GOPHER。甚至識別“mailto”,如果文件名指向“mailto:zxn@hq.cninfo.net”,它啟動電子郵件程序並打開一個待編輯的新郵件,例如:
ShellExecute(NULL, "open",“mailto:zxn@hq.cninfo.net”, NULL, NULL, SW_SHOWNORMAL);打開新郵件窗口。
總之,ShellExecute函數就是如此簡單地打開磁盤文件和Internet文件。如果將第二個參數“OPEN”改為“PRINT”或者“EXPLORE”,ShellExecute將能打印文件和打開文件夾。ShellExecute還有一個擴展函數ShellExecuteEx,所帶參數中有一個特殊的結構,功能更強,或者任何你使用的別的瀏覽器打開文件。
======================================================================================
Q: 如何打開一個應用程序
ShellExecute(this->m_hWnd,"open","calc.exe","","", SW_SHOW ); 或 ShellExecute(this->m_hWnd,"open","notepad.exe","c:\\MyLog.log","",SW_SHOW ); 正如您所看到的,我並沒有傳遞程序的完整路徑。
Q: 如何打開一個同系統程序相關連的文檔?ShellExecute(this->m_hWnd,"open","c:\\abc.txt","","",SW_SHOW );
Q: 如何打開一個網頁?
ShellExecute(this->m_hWnd,"open","http://www.google.com","","", SW_SHOW );
Q: 如何激活相關程序,發送EMAIL?
ShellExecute(this->m_hWnd,"open","mailto:nishinapp@yahoo.com","","", SW_SHOW );
Q: 如何用系統打印機打印文檔?
ShellExecute(this->m_hWnd,"print","c:\\abc.txt","","", SW_HIDE);
Q: 如何用系統查找功能來查找指定文件?
ShellExecute(m_hWnd,"find","d:\\nish",NULL,NULL,SW_SHOW);
Q: 如何啟動一個程序,直到它運行結束?
SHELLEXECUTEINFO ShExecInfo = {0};
ShExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
ShExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS;
ShExecInfo.hwnd = NULL;
ShExecInfo.lpVerb = NULL;
ShExecInfo.lpFile = "c:\\MyProgram.exe";
ShExecInfo.lpParameters = "";
ShExecInfo.lpDirectory = NULL;
ShExecInfo.nShow = SW_SHOW;
ShExecInfo.hInstApp = NULL;
ShellExecuteEx(&ShExecInfo);
WaitForSingleObject(ShExecInfo.hProcess,INFINITE);
或:
PROCESS_INFORMATION ProcessInfo;
STARTUPINFO StartupInfo; //This is an [in] parameter
ZeroMemory(&StartupInfo, sizeof(StartupInfo));
StartupInfo.cb = sizeof StartupInfo ; //Only compulsory field
if(CreateProcess("c:\\winnt\\notepad.exe", NULL,
NULL,NULL,FALSE,0,NULL,
NULL,&StartupInfo,&ProcessInfo))
{
WaitForSingleObject(ProcessInfo.hProcess,INFINITE);
CloseHandle(ProcessInfo.hThread);
CloseHandle(ProcessInfo.hProcess);
}
else
{
MessageBox("The process could not be started...");
}
Q: 如何顯示文件或文件夾的屬性?
SHELLEXECUTEINFO ShExecInfo ={0};
ShExecInfo.cbSize = sizeof(SHELLEXECUTEINFO);
ShExecInfo.fMask = SEE_MASK_INVOKEIDLIST ;
ShExecInfo.hwnd = NULL;
ShExecInfo.lpVerb = "properties";
ShExecInfo.lpFile = "c:\\"; //can be a file as well
ShExecInfo.lpParameters = "";
ShExecInfo.lpDirectory = NULL;
ShExecInfo.nShow = SW_SHOW;
ShExecInfo.hInstApp = NULL;
ShellExecuteEx(&ShExecInfo);