C++程序中調用其他exe可執行文件方法


在編程過程中有個需求,點擊某個按鈕需要彈出系統的聲音控制面板。在網上查了下代碼中調用其他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中調用其他可執行程序,就我目前所知有三種方法:

  1. 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
  2. UINT WinExec(
                              LPCSTR
     lpCmdLine,  // command line
                               UINT uCmdShow           // window style
    );

    參數lpCmdLine為文件名+參數
    如果執行正確,返回值大於31

    聲明在頭文件windows.h,包含在kernel32.lib中
  3. 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中調用其他可執行程序,就我目前所知有三種方法:

  1. 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
  2. UINT WinExec(
                              LPCSTR
     lpCmdLine,  // command line
                               UINT uCmdShow           // window style
    );

    參數lpCmdLine為文件名+參數
    如果執行正確,返回值大於31

    聲明在頭文件windows.h,包含在kernel32.lib中
  3. 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);


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM