一、問題的產生
我們或多或少都有這樣的經歷,在Windows上運行的應用程序常常會異常終止,需要通過手工重新將其啟動起來。若計算機無人看守,異常終止的進程不能實時啟動,則可能給生產造成損失。
本人在開發GPS全球衛星定位系統控制中心程序時,就遇到過控制中心程序異常終止死亡的情況,由此,找出了一個自動復活死亡進程的方法,供參考。
二、相關知識
通常,把一個應用程序的一次運行實例叫做一個進程,在一個進程內又可包含多條可並發執行的路徑,每條執行路徑叫做一個線程,一個進程至少包含一個主線程。主線程負責執行運行的啟動代碼。另外,一個進程可以創建若干子進程。當進程被創建時,系統自動產生主線程,主線程然後可創建更多的線程。
我們可以編寫一個程序,讓其創建、啟動子進程,並監視進程的運行情況,在其出現異常終止時,立即重新創建並啟動子進程即可。
三、相關函數
1、創建一個子進程函數:
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
);
參數說明:
lpApplicationName:新進程將要使用的可執行文件的名字,必須包含擴展名。
LpCommandLine:新進程的命令行。若lpApplicationName為NULL,LpCommandLine 的第一個參數是新進程將要使用的可執行文件的名字,可以不包含擴展名,系統假定是exe文件。
LpProcessAttributes和lpThreadAttributes:分別是給進程對象和線程對象指定的安全屬性。
BInheritHandles:指定該進程是否繼承其父進程中的句柄。
dwCreationFlags:指定新進程產生方式的標志,可用邏輯操作符or相連接。
LpEnvironment:指向含有新進程將要使用的環境塊字符串的一塊內存,一般為NULL,使子進程繼承父進程的一組環境塊。
LpCurrentDirectory:設置子進程的當前驅動器和工作目錄, 為NULL,子進程繼承父進程的當前驅動器和工作目錄。
LpStartupInfo:指向STARTUPINFO 的結構。一般讓子進程使用缺省值。但要把該結構中的所有成員初始化為0,並設置cb為結構大小。
STARTUPINFO 結構如下:
typedef struct _STARTUPINFO {
DWord cb;
LPTSTR lpReserved;
LPTSTR lpDesktop;
LPTSTR lpTitle;
DWord dwX;
DWord dwY;
DWord dwXSize;
DWord dwYSize;
DWord dwXCountChars;
DWord dwYCountChars;
DWord dwFillAttribute;
DWord dwFlags;
Word wShowWindow;
Word cbReserved2;
LPBYTE lpReserved2;
HANDLE hStdInput;
HANDLE hStdOutput;
HANDLE hStdError;
} STARTUPINFO, *LPSTARTUPINFO;
lpProcessInformation 參數指向LPPROCESS_INFORMATION結構,CreateProcess在返回之前,填入有關子進程的信息,父進程正是利用該信息監測子進程是否終止。該結構如下:
typedef struct _PROCESS_INFORMATION {
HANDLE hProcess;
HANDLE hThread;
DWord dwProcessId;
DWord dwThreadId;
} PROCESS_INFORMATION;
hProcess和hThread分別是子進程的句柄和子進程的主線程的句柄,dwProcessId和dwThreadId分別是子進程的標識號和子進程的主線程的標識號。
2、子進程終止檢測函數
GetEXitCodeProcess(HANDLE hProcess, LPDWord lpExitCode );
Hprocess:進程句柄,lpExitCode:進程終止時的退出碼。
如果一個進程沒有終止,lpExitCode 的返回值是STILL_ACTIVE,否則返回其他值。
四、方法的Delphi5語言實現
1、創建一個新的項目 Project1
選擇File,New Application。在表單Form1上放一Memo組件,一個OK按鈕組件,改變OK按鈕組件的Cation屬性為 CreateProcess。再放一個timer組件。設置timer組件的Interval值為1000,每秒檢查一次進程是否終止。
2、在Unit1 Use節的Type後定義一個過程
procedure EstablishProcess;
在Unit1 Use節的Var後定義一個變量:
piProcInfoGPS:PROCESS_INFORMATION;
3、在Unit1 implementation節中編寫EstablishProcess過程的實現代碼如下:
procedure EstablishProcess;
Var
siStartupInfo:STARTUPINFO;
saProcess,saThread:SECURITY_ATTRIBUTES;
fSuccess:boolean;
begin
fSuccess:=false;
ZeroMemory(@siStartupInfo,sizeof(siStartupInfo));
siStartupInfo.cb:=sizeof(siStartupInfo);
saProcess.nLength:=sizeof(saProcess);
saProcess.lpSecurityDescriptor:=PChar(nil);
saProcess.bInheritHandle:=true;
saThread.nLength:=sizeof(saThread);
saThread.lpSecurityDescriptor:=PChar(nil);
saThread.bInheritHandle:=true;
fSuccess:=CreateProcess(PChar(nil),'c:sr350Sr350buff',@saProcess,@saThread,false,
CREATE_DEFAULT_ERROR_MODE,Pchar(nil),Pchar(nil),siStartupInfo,piProcInfoGPS);
if( not fSuccess)then
Form1.Memo1.Lines.Add('Create Process Sr350buff fail.')
else
Form1.Memo1.Lines.Add('Create Process Sr350buff success.')
end;
4、在CreateProcess按鈕的OnClick事件中調用過程
EstablishProcess;
5、為Timer1的OnTimer事件編寫代碼:
Procedure TForm1.Timer1Timer(Sender: TObject);
Var
dwExitCode:DWord;
fprocessExit:boolean;
Begin
dwExitCode:=0;
fprocessExit:=false;
fprocessExit:=GetExitCodeProcess(piProcInfoGPS.hProcess,dwExitCode);
if(fprocessExit and (dwExitCode<>STILL_ACTIVE))then
begin
Memo1.Lines.Add('SR350buff.exe進程終止');
CloseHandle(piProcInfoGPS.hThread);
CloseHandle(piProcInfoGPS.hProcess);
EstablishProcess;
end;
End;
6、程序中設可執行文件名為c:sr350sr350buff.exe,所以c:盤sr350目錄下需有sr350buff.exe文件。
7、編譯聯接,運行project1,單擊CreateProcess可見c:sr350sr350buff.exe啟動。關掉sr350buff.exe進程,可見sr350buff.exe自動再啟動。