1 引言
新一代操作系統(tǒng)—— Windows 2000 是一個(gè)具有完整功能的環(huán)境,與其他操作系統(tǒng)相比完成同樣的任務(wù) Windows 2000 更出色。編程人員要編寫系統(tǒng)層次應(yīng)用程序和用戶應(yīng)用程序掌握 Windows 2000 中的進(jìn)程管理至關(guān)重要。下面對 Windows 2000 進(jìn)程的創(chuàng)建作詳細(xì)論述。以便使讀者從中掌握 Windows 2000 的特性,編寫更有效和更有用的程序。
2 使用 WinExec 命令
⑴ 函數(shù)原型:
UINT Win Exec(LPCSTR lpCmdLine, UINT uCmdShow);
⑵ 參數(shù):
lpCmdLine :指向一個(gè)空結(jié)束的字符串,串中包含將要執(zhí)行的應(yīng)用程序的命令行(文件名加上可選參數(shù))。
uCmdShow :定義 Windows 應(yīng)用程序的窗口如何顯示,并為 CreateProcess 函數(shù)提供 STARTUPINFO 參數(shù)的 wShowWindow 成員的值。
⑶ 返回值:
若函數(shù)調(diào)用成功,則返回值大于 31 。若函數(shù)調(diào)用失敗,則返回值為下列之一:
① 0 :系統(tǒng)內(nèi)存或資源已耗盡。
② ERROR_BAD_FORMAT : EXE 文件無效(非 Win32.EXE 或 .EXE 影像錯(cuò)誤)。
③ ERROR_FILE_NOT_FOUND :指定的文件未找到。
④ ERROR_PATH_NOT_FOUND :指定的路徑未找到。
雖然 Microsoft 認(rèn)為 WinExec 已過時(shí),但是在許多時(shí)候,簡單的 WinExec 函數(shù)仍是運(yùn)行新程序的最好方式。簡單地傳送作為第一個(gè)參數(shù)的命令行,還需要決定如何顯示程序(該程序也許會忽視它)的第二個(gè)參數(shù)。通常,將其設(shè)置為 SW_SHOW ,也可嘗試 SW_MINIMIZED 或 SW_MAXIMIZED 。 WinExec 不允許用 CreateProcess 獲得的所有選項(xiàng),而它的確簡單。
3 使用 ShellExecute 命令
⑴ 函數(shù)原型:
HINSTANCE ShellExecute(HWND hwnd, LPCTSTR lpOperation, LPCTSTR lpFile, LPCTSTR lpParameters, LPCTSTR lpDirectory, INT nShowCmd);
⑵ 參數(shù):
hwnd :指向父窗口的窗口句柄。此窗口接收應(yīng)用程序產(chǎn)生的任何信息框。
lpOperation :一個(gè)空結(jié)束的字符串地址,此字符串指定要執(zhí)行的操作。下面的操作字符串是有效的:
"open"
此函數(shù)打開由參數(shù) lpFile 指定的文件,此文件可以是一個(gè)可執(zhí)行文件或文檔文件,也可是一個(gè)文件夾。
"print"
此函數(shù)打印由參數(shù) lpFile 指定的文件,此文件應(yīng)是一個(gè)文檔文件,假如此文件是一個(gè)可執(zhí)行文件,則打開此文件。
"explore"
此函數(shù)搜索由參數(shù) lpFile 指定的文件夾,此文件應(yīng)是一個(gè)文檔文件,
此參數(shù)可以為空。這種情況下,函數(shù)用于打開由參數(shù) lpFile 指定的文件。
lpFile :一個(gè)空結(jié)束的字符串地址,此字符串指定要打開或打印的文件或者是要打開或搜索的文件夾。
lpParameters :假如參數(shù) lpFile 指定一個(gè)可執(zhí)行文件, lpParameters 則是一個(gè)空結(jié)束的字符串地址,此字符串指定要傳遞給應(yīng)用程序的參數(shù)。假如 lpFile 指定一個(gè)文檔文件, lpParameters 應(yīng)為空。
lpDirectory :一個(gè)空結(jié)束的字符串地址,此字符串指定默認(rèn)目錄。
nShowCmd :假如 lpFile 指定一個(gè)可執(zhí)行文件, nShowCmd 表明應(yīng)用程序打開時(shí)如何顯示。假如 lpFile 指定一個(gè)文檔文件, nShowCmd 應(yīng)為空。
⑶ 返回值:
若函數(shù)調(diào)用成功,則返回值大于 32 ,否則為一個(gè)小于等于 32 的錯(cuò)誤值。
說明:可以用此函數(shù)打開或搜索一個(gè)外殼文件夾。打開文件夾可用下面任何一種形式:
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 命令雖已過時(shí)但易于得到。該命令向命令解釋程序提出打開、瀏覽或打印文檔或文件夾的請求,雖然可以用 ShellExecute 運(yùn)行程序,但通常只發(fā)送文檔名,而命令解釋程序則決定要運(yùn)行那個(gè)程序。另外在打開目錄文件夾時(shí), ShellExecute 命令非常有用。
⑷ 程序示例
下面通過一個(gè)例子來說名 WinExec 和 ShellExecute 的使用。下面程序有控制臺程序示例,其使用兩種不同的方法,打開文本文件。下面程序使用 WinExec ,并明確指定使用記事本程序。然后,使用 ShellExecute, 打開文本文件。
程序清單
#include <windows.h>
#include <iostream.h>
void main(int argc,char *argv[])
{
cout<<”O(jiān)pening 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<<”O(jiān)pening with ShellExecute\n”;
if (ShellExecute (NULL,”open”,
”readme.txt”,NULL,NULL,SW_SHOW)<(HANDLE) 32)
MessagBox(NULL,”Can‘t ShellExecute\n”,NULL,MB_OK);
}
4 使用 CreateProcess 命令
⑴ 函數(shù)原型:
BOOL CreateProcess(
LPCTSTR lpApplicationName ,
LPTSTR lpCommandLine ,
LPSECURITY_ATTRIBUTES lpProcessAttributes ,
LPSECURITY_ATTRIBUTES lpThreadAttributes ,
BOOL bInheritHandles ,
DWORD dwCreationFlags ,
LPVOID lpEnvironment ,
LPCTSTR lpCurrentDirectory ,
LPSTARTUPINFO lpStartupInfo ,
LPPROCESS_INFORMATION lpProcessInformation
);
⑵ 參數(shù):
lpApplicationName :指向一個(gè)以空結(jié)尾的串,他指定了要執(zhí)行的模塊
lpCommandLine :指向一個(gè)以空結(jié)尾的串,該串定義了要執(zhí)行的命令行。
lpProcessAttributes :指向一個(gè) SECURITY_ATTRIBUTES 結(jié)構(gòu),該結(jié)構(gòu)決定了返回的句柄是否可被子進(jìn)程繼承。
lpThreadAttributes :指向一個(gè) SECURITY_ATTRIBUTES 結(jié)構(gòu),該結(jié)構(gòu)決定了返回的句柄是否可被子進(jìn)程繼承。
bInheritHandles , : 表明新進(jìn)程是否從調(diào)用進(jìn)程繼承句柄。
dwCreationFlags : 定義控制優(yōu)先類和進(jìn)程創(chuàng)建的附加標(biāo)志。
lpEnvironment :指向一個(gè)新進(jìn)程的環(huán)境塊。
lpCurrentDirectory :指向一個(gè)以空結(jié)尾的串,該串定義了子進(jìn)程的當(dāng)前驅(qū)動(dòng)器和當(dāng)前目錄。
lpStartupInfo :指向一個(gè) STARTUPINFO 結(jié)構(gòu),該結(jié)構(gòu)定義了新進(jìn)程的主窗口將如何顯示。
lpProcessInformation : 指向 PROCESS_INFORMATION 結(jié)構(gòu),該結(jié)構(gòu)接受關(guān)于新進(jìn)程的表示信息。
⑶ 返回值:
若函數(shù)調(diào)用成功,則返回值不為 0 ;若函數(shù)調(diào)用失敗,返回值為 0 。
ShellExecute 和 WinExec 命令用于簡單的作業(yè)。如果要完全控制一個(gè)新進(jìn)程,就必須調(diào)用 CreateProcess 。
在上述參數(shù)中,參數(shù) lpStartupInfo 是 STARTUPINFO 結(jié)構(gòu)??梢杂脕碓O(shè)置控臺的標(biāo)題,新窗口的的初始大小和位置,及重定向標(biāo)準(zhǔn)輸入和輸出。新程序通常可以忽略多數(shù)這些數(shù)據(jù)項(xiàng),如果選擇那樣做的話。可以規(guī)定該結(jié)構(gòu)體中的標(biāo)志,已表明要設(shè)置的數(shù)據(jù)段。有時(shí),不想設(shè)置任何信息,也必須傳遞一個(gè)有效的指針給空結(jié)構(gòu)(確定設(shè)置大小到 cb ,及設(shè)置 dwFlags 成員為 0 )。參數(shù) lpProcessInformation 返回進(jìn)程和線程句柄,還包括進(jìn)程和線程 ID 。這些句柄擁有在參數(shù) lpProcessAttributes 和 lpThreadAttributes 中規(guī)定的訪問。
要注意,針對 CreateProcess 的一些參數(shù)對控制臺應(yīng)用程序是特定的,而其它參數(shù)則對各種應(yīng)用程序有用。大多數(shù)情況下,并不一定要填入 STARTUPINFO 結(jié)構(gòu),但無論如何必須提供它。其返回值是布爾型的,而真正感興趣的返回值發(fā)生于作為參數(shù)傳送的結(jié)構(gòu)中( PROCESS_INFORMATION )。 CreateProcess 返回該結(jié)構(gòu)中的進(jìn)程 ID 及其句柄,以及初始線程 ID 及其句柄??梢詫?ID 發(fā)送到其它進(jìn)程,或使用句柄來控制新進(jìn)程。
⑷ 相關(guān)命令
給定進(jìn)程句柄,就可以用相關(guān)命令來控制進(jìn)程。下面我們討論進(jìn)程結(jié)束的確定,進(jìn)程結(jié)束的確定有以下幾種方法:
① 調(diào)用 GetExitCodeProcess
命令 GetExitCodeProcess 既能返回 STILL_ACTIVE ,也能返回進(jìn)程退出值(如果進(jìn)程結(jié)束時(shí))返回值需要一個(gè)指針,其指向命令填充的變量。
② 調(diào)用 WaitForSingleObject
WaitForSingleObject 的目的是要確定句柄是否處于發(fā)送信號的狀態(tài)。當(dāng)進(jìn)程結(jié)束時(shí),進(jìn)程句柄發(fā)出信號。當(dāng)調(diào)用 WaitForSingleObject 時(shí),就規(guī)定進(jìn)程句柄和超時(shí)值,如果超時(shí)為 0 ,則該命令就立刻返回,且能夠確定進(jìn)程的狀態(tài)。如果超時(shí)是常數(shù) INFINITE ,則命令就不返回,直到目標(biāo)進(jìn)程退出為止。當(dāng)然,還可以規(guī)定超時(shí)值,其導(dǎo)致該命令等待要結(jié)束的進(jìn)程一段時(shí)間。如果進(jìn)程在超時(shí)屆滿前結(jié)束,該命令就返回,并指出句柄在發(fā)射信號狀態(tài)。否則,就返回一個(gè)負(fù)值。不管句柄在何種狀態(tài), WaitForSingleObject 將成功返回,沒有錯(cuò)誤發(fā)生。要確定進(jìn)程的狀態(tài),就必須比較返回值為 WAIT_OBJECT_0 (已發(fā)信號的)和 WAIT_TIMEOUT (未發(fā)信號的)。真正的錯(cuò)誤返回值為 WAIT_FAILED 。另外可能的返回值是 WAIT_ABANDONED ,是不會看到何時(shí)處理進(jìn)程。要等待一個(gè)進(jìn)程,就必須帶有 SYNCHRONIZE 特權(quán)的打開局柄。
這里要注意,進(jìn)程 ID 與進(jìn)程句柄不同。不能簡單地在進(jìn)程之間傳送句柄,這意味著除非有句柄,否則不能從外部進(jìn)程直接操縱一個(gè)進(jìn)程。不過 OpenProcess 命令將允許任何程序(有足夠的安全特權(quán))將進(jìn)程標(biāo)示符(可以用來于其它進(jìn)程通信)變換為進(jìn)程句柄。通過調(diào)用 GetCurrentProcessId ,還可以了解當(dāng)前進(jìn)程標(biāo)示符。如果要想與其他無關(guān)的進(jìn)程共享,以使能夠打開進(jìn)程句柄,這是非常有用的。但調(diào)用 OpenProcess 時(shí),可以請求對進(jìn)程的訪問。對每種進(jìn)程的訪問,也許有或也許沒有訪問要打開進(jìn)程的安全性,于是試圖請求是僅僅需要的。例如,如果要了解進(jìn)程的返回代碼,就需要 PROCESS_QUERY_INFORMATION 的訪問。要終止進(jìn)程,就必須有 PROCESS_TERMINATE 的訪問。
⑸ 程序示例
下面通過一個(gè)例子來說名 CreateProcess 和相關(guān)命令的使用。下面程序是兩個(gè)簡單的控制臺應(yīng)用程序,第一個(gè)程序( MASTER )運(yùn)行第二個(gè)程序( SLAVE ) , 并進(jìn)入睡眠。 SLAVE 程序從命令行讀取源程序的進(jìn)程 ID(PID), 并等待 MASTER 程序終止。這些程序說明了以下幾個(gè)重要技術(shù):
l 使用 CreateProcess
l 使用 OpenProcess
l 使用 WaitForSingleObject
程序清單 MASTER 程序
#include <windows.h>
#include <iostream.h>
#include <stdio.h>
#include <string.h>
void main(int argc,char *argv[])
{
char cmd[128];
if (argc!=1)
strcpy(cmd,argv[1]);
else
strcpy(cmd,”slave.exe”);
int pid=GetCurrentProcessId();
sprintf(cmd+strlen(cmd),” %d”,pid);
cout<<”Master: Starting:”<<cmd<<”\n”);
cout.flush();
STARTUPINFO info;
memset(&info,0,sizeof(info));
info.cb=sizeof(info);
PROCESS_INFORMATION pinfo;
If(!CreateProcess(NULL,cmd,NULL,NULL,FALSE<NORMAL_PRIORITY_CLASS,NULL,NULL,&info,&pinfo))
{
c out<<”Master:Slave process did not start\n”;
c out<<” Master:Try naming slave process on the command line\n”;
}
cout<<”Master:Sleeping\n”;
cout.flush();
Sleep(15000);
Cout<<”Master:Exiting\n”;
exit(0);
}
程序清單 SLAVE 程序
#include <window.h>
#include <iostream.h>
#include <stdio.h>
void main(int argc,char *argv[])
{
if (argc!=2)
{
cout<<”Slave:Please rrun MASTER.EXE instead.\n”;
exit(1);
}
int pid=atoi(argv[1]);
HANDLE process=OpenProcess(PROCESS_QUERY_INFORMATION|SYNCHRONIZE,FALSE,pid);
if (!process) cot<<”Slave:Error opening process\n”;
cout<<”Slave :Waiting for master to finish\n”;
cout.flush();
if (WaitForSingleObject(process,INFINITE)==STAUTE_WAIT_0)
cout<<”Slave:Master completed\n”;
else
cout<<”Slave:Unexpected error\n”;
exit(0);
}
5 小結(jié):
通過本文介紹,讀者應(yīng)能掌握 Windows 2000 進(jìn)程的創(chuàng)建,了解 Windows 2000 進(jìn)程的控制。
參考文獻(xiàn)
1 《新編 WINDOWS API 參考大全》朱友芹 主編 ………… 電子工業(yè)出版社, 2000 。
2 《 Windows 2000 系統(tǒng)編程》 ( 美 ) AI Williams 著 鈕文良等譯 … 機(jī)械工業(yè)出版社, 2001 。