您好,登錄后才能下訂單哦!
c#中怎么實現守護進程,針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。
在配置檔中配置你要守護的應用程序,應用程序之間用逗號隔開:
<?xml version="1.0" encoding="utf-8" ?><configuration> <appSettings> <add key="ProcessAddress" value=" d:\war3.exe, d:\note.txt, d:\girl.jpg " /> </appSettings></configuration>
該項目是Windows服務,直接打開“CocoWatcher.exe”會報錯,如圖1所示:
運行該守護程序需安裝Windows服務,點擊批處理文檔“安裝.bat”即可,“安裝.bat”具體內容如下:
"%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe"net start "CocoWatcher"pause
如果你想卸載該守護程序,點擊批處理文檔“卸載.bat”,“卸載.bat”具體內容如下:
net stop "CocoWatcher""%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe" -utaskkill /f /im CocoWatcher.exepause
3. 需求分析
用戶指定要守護的應用程序(數量不限),該應用程序不僅包括exe可執行文件,還包括諸如jpg、txt等所有能雙擊打開執行的應用程序。用戶設定好要守護的應用程序后,關閉應用程序(包括合法和非法關閉),該應用程序要能立即重啟打開。當電腦重啟時,要守護的應用程序也能自動全部打開。
4. 詳細設計
要實現上述需求,首先要提供一個配置檔,讓用戶能隨意配置要守護的應用程序。那么,該配置檔要配置應用程序的什么信息呢?答案:應用程序的全路徑。
好,我們已經知道了要守護的應用程序的全路徑,接下來怎樣完成守護任務呢?首先,我們應該打開任務管理器,查看一下正在運行的有哪些進程,然后逐一讀取出這些進程的全路徑,與要守護的應用程序的全路徑比對,如果一致,說明要守護的應用程序已開啟了,此時要分配一條線程監控該進程句柄,當該進程句柄返回信息,說明該進程已關閉,此時釋放進程句柄內存,并重啟該進程。如果遍歷任務管理進程列表中所有進程,沒有找到與要守護的應用程序的全路徑一致的進程,說明要守護的應用程序尚未打開,此時要啟動該應用程序,然后轉入監控流程。
值得注意的是,一定要額外分配線程去監控要守護的應用程序,為什么?因為如果你用主線程(入口函數線程)去執行監控任務,會被長期阻塞,直到進程退出才會被激活,這樣就無法運行后續程序。況且,監控程序要實現持續監控,要使用死循環,如果主線程進入死循環,就無法監控其他要守護的進程了。
5. 代碼詳解
Windows服務的開發步驟,請參考MSDN,此處略去。下面將關鍵代碼貼出,加以解釋。
讀取配置檔中“ProcessAddress”節點,獲取要守護的應用程序全目錄,驗證應用程序全目錄,如果合法,進入掃描任務管理器進程列表流程。
/// <summary>/// 開始監控/// </summary>private void StartWatch(){ if (this._processAddress != null) { if (this._processAddress.Length > 0) { foreach (string str in _processAddress) { if (str.Trim() != "") { if (File.Exists(str.Trim())) { this.ScanProcessList(str.Trim()); } } } } }}
打開任務管理器,查看一下正在運行的有哪些進程,然后逐一讀取出這些進程的全路徑,與要守護的應用程序的全路徑比對,如果一致,說明要守護的應用程序已開啟了,進入監控流程。如果遍歷任務管理進程列表中所有進程,沒有找到與要守護的應用程序的全路徑一致的進程,說明要守護的應用程序尚未打開,此時要啟動該應用程序,然后轉入監控流程。
/// <summary>/// 掃描進程列表,判斷進程對應的全路徑是否與指定路徑一致/// 如果一致,說明進程已啟動/// 如果不一致,說明進程尚未啟動/// </summary>/// <param name="strAddress"></param>private void ScanProcessList(string address){ Process[] arrayProcess = Process.GetProcesses(); foreach (Process p in arrayProcess) { //System、Idle進程會拒絕訪問其全路徑 if (p.ProcessName != "System" && p.ProcessName != "Idle") { try { if (this.FormatPath(address) == this.FormatPath(p.MainModule.FileName.ToString())) { //進程已啟動 this.WatchProcess(p, address); return; } } catch { //拒絕訪問進程的全路徑 this.SaveLog("進程(" + p.Id.ToString() + ")(" + p.ProcessName.ToString() + ")拒絕訪問全路徑!"); } } } //進程尚未啟動 Process process = new Process(); process.StartInfo.FileName = address; process.Start(); this.WatchProcess(process, address);}
分配一條線程,執行監控任務:
/// <summary> /// 監聽進程 /// </summary> /// <param name="p"></param> /// <param name="address"></param> private void WatchProcess(Process process, string address) { ProcessRestart objProcessRestart = new ProcessRestart(process, address); Thread thread = new Thread(new ThreadStart(objProcessRestart.RestartProcess)); thread.Start(); } public class ProcessRestart{ //字段 private Process _process; private string _address; /// <summary> /// 構造函數 /// </summary> public ProcessRestart() {} /// <summary> /// 構造函數 /// </summary> /// <param name="process"></param> /// <param name="address"></param> public ProcessRestart(Process process, string address) { this._process = process; this._address = address; } /// <summary> /// 重啟進程 /// </summary> public void RestartProcess() { try { while (true) { this._process.WaitForExit(); this._process.Close(); //釋放已退出進程的句柄 this._process.StartInfo.FileName = this._address; this._process.Start(); Thread.Sleep(1000); } } catch (Exception ex) { ProcessWatcher objProcessWatcher = new ProcessWatcher(); objProcessWatcher.SaveLog("RestartProcess() 出錯,監控程序已取消對進程(" + this._process.Id.ToString() +")(" + this._process.ProcessName.ToString() + ")的監控,錯誤描述為:" + ex.Message.ToString()); } }}
關于c#中怎么實現守護進程問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。