vc++ 在程序中運行另一個程序的方法

在vc++ 程序中運行另一個程序的方法有三個: WinExec(),ShellExcute()和CreateProcess()

三個SDK函數: WinExec,ShellExecute ,CreateProcess可以實現調用其他程序的要求,其中以WinExec最爲簡單,ShellExecute比WinExec靈活一些,CreateProcess最爲複雜。
    WinExec 兩個參數,前一個指定路徑,後一個指定顯示方式。
    ShellExecute 可以指定工作目錄,並且還可以尋找文件的關聯直接打開不用加載與文件關聯的應用程序,ShellExecute還可以打開網頁,啓動相應的郵件關聯發送郵件等等。
    CreateProcess 一共有十個參數,不過大部分都可以用NULL代替,它可以指定進程的安全屬性,繼承信息,類的優先級等等。如果我們要得到足夠多的關於新的進程的信息,控制新的進程的細節屬性,若要達到這些目的,我們就需要使用CreateProcess函數了。
 
三個SDK函數( WinExec、ShellExec、CrateProcess )的語法:
 

WinExec

這個函數最簡單,只有兩個參數,原型如下:

 
      UINT WinExec(

      LPCSTR lpCmdLine,   // 命令路徑

      UINT uCmdShow      // 顯示方式

      
;

  lpCmdLine:指向一個空結束的字符串,串中包含將要執行的應用程序的命令行(文件名加上可選參數)。

  uCmdShow:定義Windows應用程序的窗口如何顯示,併爲CreateProcess函數提供STARTUPINFO參數的wShowWindow成員的值。

  返回值:

  若函數調用成功,則返回值大於31。若函數調用失敗,則返回值爲下列之一:

  ① 0:系統內存或資源已耗盡。

  ② ERROR_BAD_FORMAT:EXE文件無效(非Win32.EXE或.EXE影像錯誤)。

  ③ ERROR_FILE_NOT_FOUND:指定的文件未找到。

  ④ ERROR_PATH_NOT_FOUND:指定的路徑未找到。

  雖然Microsoft認爲WinExec已過時,但是在許多時候,簡單的WinExec函數仍是運行新程序的最好方式。簡單地傳送作爲第一個參數的命令行,還需要決定如何顯示程序(該程序也許會忽視它)的第二個參數。通常,將其設置爲SW_SHOW,也可嘗試SW_MINIMIZED或SW_MAXIMIZED。WinExec不允許用CreateProcess獲得的所有選項,而它的確簡單。


使用方法如下:

WinExec("Notepad.exe", SW_SHOW);  // 打開記事本
WinExec("D://Program Files//Test//Test.exe",SW_SHOWMAXIMIZED); // 以最大化的方式打Test.exe其中這裏的SW_SHOW,SW_SHOWMAXIMIZED都是執行程序時窗口的顯示方式,在winuser.h中定義。
需要注意的是若用 SW_SHOWMAXMIZED 方式去加載一個無最大化按鈕的程序,譬如Neterm,Calc 等等,就不會出現正常的 窗體,但是已經被加到任務列表裏了。
 

ShellExecute

原型如下:

      HINSTANCE ShellExecute(

      HWND hwnd,           //父窗口句柄

      LPCTSTR lpOperation,   //操作, 打開式 "edit","explore","open","find","print","NULL"

      LPCTSTR lpFile,         //文件名,前面可加路徑

      LPCTSTR lpParameters,   //參數

      LPCTSTR lpDirectory,    //默認文件夾

      INT nShowCmd          //顯示方式

);


參數:

  hwnd:指向父窗口的窗口句柄。此窗口接收應用程序產生的任何信息框。

  lpOperation:一個空結束的字符串地址,此字符串指定要執行的操作。下面的操作字符串是有效的:

"open" 此函數打開由參數lpFile指定的文件,此文件可以是一個可執行文件或文檔文件,也可是一個文件夾。
"print" 此函數打印由參數lpFile指定的文件,此文件應是一個文檔文件,假如此文件是一個可執行文件,則打開此文件。
"explore" 此函數搜索由參數lpFile指定的文件夾,此文件應是一個文檔文件,

   此參數可以爲空。這種情況下,函數用於打開由參數lpFile指定的文件。

  lpFile:一個空結束的字符串地址,此字符串指定要打開或打印的文件或者是要打開或搜索的文件夾。

  lpParameters:假如參數lpFile指定一個可執行文件,lpParameters則是一個空結束的字符串地址,此字符串指定要傳遞給應用程序的參數。假如lpFile指定一個文檔文件,lpParameters應爲空。

  lpDirectory:一個空結束的字符串地址,此字符串指定默認目錄。

  nShowCmd:假如lpFile指定一個可執行文件,nShowCmd表明應用程序打開時如何顯示。假如lpFile指定一個文檔文件,nShowCmd應爲空。

 返回值:

  若函數調用成功,則返回值大於32,否則爲一個小於等於32的錯誤值。

  說明:可以用此函數打開或搜索一個外殼文件夾。打開文件夾可用下面任何一種形式:

ShellExecute(handle, NULL, path_to_folder, NULL, NULL, SW_SHOWNORMAL);
  
  或

ShellExecute(handle, "open", path_to_folder, NULL, NULL, SW_SHOWNORMAL);

  搜索文件夾,可用如下形式

ShellExecute(handle, "explore", path_to_folder, NULL, NULL, SW_SHOWNORMAL);

  ShellExecute命令雖已過時但易於得到。該命令向命令解釋程序提出打開、瀏覽或打印文檔或文件夾的請求,雖然可以用ShellExecute運行程序,但通常只發送文檔名,而命令解釋程序則決定要運行那個程序。另外在打開目錄文件夾時,ShellExecute命令非常有用。


使用方法如下:
ShellExecute(NULL,"open","C://Test.txt",NULL,NULL,SW_SHOWNORMAL); // 打開C:/Test.txt 文件
ShellExecute(NULL, "open", "
http://www.google.com", NULL, NULL, SW_SHOWNORMAL); // 打開網頁www.google.com
ShellExecute(NULL,"explore", "D://C++",NULL,NULL,SW_SHOWNORMAL); // 打開目錄D:/C++
ShellExecute(NULL,"print","C://Test.txt",NULL,NULL, SW_HIDE); // 打印文件C:/Test.txt
ShellExecute不支持定向輸出。

程序示例

  下面通過一個例子來說名WinExec和ShellExecute的使用。下面程序有控制檯程序示例,其使用兩種不同的方法,打開文本文件。下面程序使用WinExec,並明確指定使用記事本程序。然後,使用ShellExecute,打開文本文件。

  程序清單

#include <windows.h>
#include <iostream.h>

void main(int argc,char *argv[])
{
 cout<<”Opening with WinExec/n”;
 if (WinExec(“notepad readme.txt”,SH_SHOW)<32)
  MessagBox(NULL,”Can’t WinExec”,NULL,MB_OK);
  cout<<”Press Enter/n”;
  MessagBox(NULL,”Press OK to continue”,”Progrm Launched”,MB_OK);
  cout<<”Opening with ShellExecute/n”;
 if (ShellExecute (NULL,”open”,”readme.txt”,NULL,NULL,SW_SHOW)<(HANDLE) 32)
  MessagBox(NULL,”Can’t ShellExecute/n”,NULL,MB_OK);
}


 

CreateProcess

原型如下:

      BOOL CreateProcess(

      LPCTSTR lpApplicationName, //執行程序名

      LPTSTR lpCommandLine,  // 參數行

      //下面兩個參數描述了所創建的進程和線程的安全屬性,如果爲NULL則使用默認的安全屬性

      LPSECURITY_ATTRIBUTES lpProcessAttributes,  // process security attributes

      LPSECURITY_ATTRIBUTES lpThreadAttributes,   // thread security attributes

      BOOL bInheritHandles,  // 繼承標誌

      DWORD dwCreationFlags, // 創建標誌

      LPVOID lpEnvironment,  // 環境變量

      LPCTSTR lpCurrentDirectory,   // 運行該進程的初始目錄

      LPSTARTUPINFO lpStartupInfo,  // 用於在創建子進程時設置各種屬性

      LPPROCESS_INFORMATION lpProcessInformation //用於在進程創建後接受相關信息

      


使用方法如下:
             PROCESS_INFORMATION pi;
                       STARTUPINFO si;
                       memset(&si,0,sizeof(si));
                       si.cb=sizeof(si);
                      si.wShowWindow=SW_SHOW;
                       si.dwFlags=STARTF_USESHOWWINDOW;
                       bool fRet=CreateProcess("D://putty.exe",NULL,NULL,FALSE,NULL,NULL,NULL,NULL,&si,&pi);


參數:

  lpApplicationName:指向一個以空結尾的串,他指定了要執行的模塊

  lpCommandLine:指向一個以空結尾的串,該串定義了要執行的命令行。

  lpProcessAttributes:指向一個SECURITY_ATTRIBUTES結構,該結構決定了返回的句柄是否可被子進程繼承。

  lpThreadAttributes:指向一個SECURITY_ATTRIBUTES結構,該結構決定了返回的句柄是否可被子進程繼承。

  bInheritHandles,:表明新進程是否從調用進程繼承句柄。

  dwCreationFlags:定義控制優先類和進程創建的附加標誌。

  lpEnvironment:指向一個新進程的環境塊。

  lpCurrentDirectory:指向一個以空結尾的串,該串定義了子進程的當前驅動器和當前目錄。

  lpStartupInfo:指向一個STARTUPINFO結構,該結構定義了新進程的主窗口將如何顯示。

  lpProcessInformation:指向PROCESS_INFORMATION結構,該結構接受關於新進程的表示信息。

 返回值:

  若函數調用成功,則返回值不爲0;若函數調用失敗,返回值爲0。

  ShellExecute和WinExec命令用於簡單的作業。如果要完全控制一個新進程,就必須調用CreateProcess。

  在上述參數中,參數lpStartupInfo是STARTUPINFO結構。可以用來設置控臺的標題,新窗口的的初始大小和位置,及重定向標準輸入和輸出。新程序通常可以忽略多數這些數據項,如果選擇那樣做的話。可以規定該結構體中的標誌,已表明要設置的數據段。有時,不想設置任何信息,也必須傳遞一個有效的指針給空結構(確定設置大小到cb,及設置dwFlags成員爲0)。參數lpProcessInformation返回進程和線程句柄,還包括進程和線程ID。這些句柄擁有在參數lpProcessAttributes和lpThreadAttributes中規定的訪問。

要注意,針對CreateProcess的一些參數對控制檯應用程序是特定的,而其它參數則對各種應用程序有用。大多數情況下,並不一定要填入STARTUPINFO結構,但無論如何必須提供它。其返回值是布爾型的,而真正感興趣的返回值發生於作爲參數傳送的結構中(PROCESS_INFORMATION)。CreateProcess返回該結構中的進程ID及其句柄,以及初始線程ID及其句柄。可以將ID發送到其它進程,或使用句柄來控制新進程。
可以看出,通過上面的幾個不同的方法,都可以實現在應用程序中打開其他應用程序的目的,其中有些方法可能會麻煩一點,所以就需要我們根據不同的目的去選擇最適合自己的方法去實現自己的目的!

 

 

關於三個SDK函數: WinExec, ShellExecute,CreateProcess 的其他注意事項:

1、定義頭文件

在頭文件stdafx.h中必須定義以下兩個頭文件:

#include <shlobj.h> // 可替換爲 windows.h
#include <shellapi.h>
如果定義了頭文件 #include <windows.h>的話就不必定義 #include <shlobj.h>了,"windows.h" 不光是包含了"shellapi.h",它還定義了許多數據類型,如果沒有這些數據類型,shellapi.h本身會出錯。
 
2、定義路徑

C++中所表示的路徑要用 " // "而不是平常所用的" / ",所以以上三個函數表示路徑都爲:

Disk://Directory//...//File name

WinExec("D://Program Files//Test//Test.exe",SW_SHOWMAXIMIZED);

ShellExecute(NULL,"open","C://Test.txt",NULL,NULL,SW_SHOWNORMAL);
bool fRet=CreateProcess("D://putty.exe",NULL,NULL,FALSE,NULL,NULL,NULL,NULL,&si,&pi);

這個大蝦的也不錯哦:http://www.blogcn.com/u2/81/45/jinrightherewaiting/index.html

發佈了5 篇原創文章 · 獲贊 5 · 訪問量 5萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章