如何利用c#實(shí)現(xiàn)通用守護(hù)進(jìn)程
1. 下載
源碼下載:http://files.cnblogs.com/tianzhiliang/CocoWatcher.rar
安裝包下載:http://files.cnblogs.com/tianzhiliang/CocoWatcher_Setup.rar
本地下載:http://xiazai.jb51.net/201910/yuanma/CocoWatcher(jb51net).rar
2. 安裝注意事項(xiàng)
在配置檔中配置你要守護(hù)的應(yīng)用程序,應(yīng)用程序之間用逗號(hào)隔開:
<?xml version="1.0" encoding="utf-8" ?> <configuration> <appSettings> <add key="ProcessAddress" value=" d:\war3.exe, d:\note.txt, d:\girl.jpg " /> </appSettings> </configuration>
該項(xiàng)目是Windows服務(wù),直接打開“CocoWatcher.exe”會(huì)報(bào)錯(cuò),如圖1所示:
運(yùn)行該守護(hù)程序需安裝Windows服務(wù),點(diǎn)擊批處理文檔“安裝.bat”即可,“安裝.bat”具體內(nèi)容如下:
"%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe" net start "CocoWatcher" pause
如果你想卸載該守護(hù)程序,點(diǎn)擊批處理文檔“卸載.bat”,“卸載.bat”具體內(nèi)容如下:
net stop "CocoWatcher" "%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe" -u taskkill /f /im CocoWatcher.exe pause
3. 需求分析
用戶指定要守護(hù)的應(yīng)用程序(數(shù)量不限),該應(yīng)用程序不僅包括exe可執(zhí)行文件,還包括諸如jpg、txt等所有能雙擊打開執(zhí)行的應(yīng)用程序。用戶設(shè)定好要守護(hù)的應(yīng)用程序后,關(guān)閉應(yīng)用程序(包括合法和非法關(guān)閉),該應(yīng)用程序要能立即重啟打開。當(dāng)電腦重啟時(shí),要守護(hù)的應(yīng)用程序也能自動(dòng)全部打開。
4. 詳細(xì)設(shè)計(jì)
要實(shí)現(xiàn)上述需求,首先要提供一個(gè)配置檔,讓用戶能隨意配置要守護(hù)的應(yīng)用程序。那么,該配置檔要配置應(yīng)用程序的什么信息呢?答案:應(yīng)用程序的全路徑。
好,我們已經(jīng)知道了要守護(hù)的應(yīng)用程序的全路徑,接下來怎樣完成守護(hù)任務(wù)呢?首先,我們應(yīng)該打開任務(wù)管理器,查看一下正在運(yùn)行的有哪些進(jìn)程,然后逐一讀取出這些進(jìn)程的全路徑,與要守護(hù)的應(yīng)用程序的全路徑比對(duì),如果一致,說明要守護(hù)的應(yīng)用程序已開啟了,此時(shí)要分配一條線程監(jiān)控該進(jìn)程句柄,當(dāng)該進(jìn)程句柄返回信息,說明該進(jìn)程已關(guān)閉,此時(shí)釋放進(jìn)程句柄內(nèi)存,并重啟該進(jìn)程。如果遍歷任務(wù)管理進(jìn)程列表中所有進(jìn)程,沒有找到與要守護(hù)的應(yīng)用程序的全路徑一致的進(jìn)程,說明要守護(hù)的應(yīng)用程序尚未打開,此時(shí)要啟動(dòng)該應(yīng)用程序,然后轉(zhuǎn)入監(jiān)控流程。
值得注意的是,一定要額外分配線程去監(jiān)控要守護(hù)的應(yīng)用程序,為什么?因?yàn)槿绻阌弥骶€程(入口函數(shù)線程)去執(zhí)行監(jiān)控任務(wù),會(huì)被長(zhǎng)期阻塞,直到進(jìn)程退出才會(huì)被激活,這樣就無法運(yùn)行后續(xù)程序。況且,監(jiān)控程序要實(shí)現(xiàn)持續(xù)監(jiān)控,要使用死循環(huán),如果主線程進(jìn)入死循環(huán),就無法監(jiān)控其他要守護(hù)的進(jìn)程了。
5. 代碼詳解
Windows服務(wù)的開發(fā)步驟,請(qǐng)參考MSDN,此處略去。下面將關(guān)鍵代碼貼出,加以解釋。
讀取配置檔中“ProcessAddress”節(jié)點(diǎn),獲取要守護(hù)的應(yīng)用程序全目錄,驗(yàn)證應(yīng)用程序全目錄,如果合法,進(jìn)入掃描任務(wù)管理器進(jìn)程列表流程。
/// <summary> /// 開始監(jiān)控 /// </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()); } } } } } }
打開任務(wù)管理器,查看一下正在運(yùn)行的有哪些進(jìn)程,然后逐一讀取出這些進(jìn)程的全路徑,與要守護(hù)的應(yīng)用程序的全路徑比對(duì),如果一致,說明要守護(hù)的應(yīng)用程序已開啟了,進(jìn)入監(jiān)控流程。如果遍歷任務(wù)管理進(jìn)程列表中所有進(jìn)程,沒有找到與要守護(hù)的應(yīng)用程序的全路徑一致的進(jìn)程,說明要守護(hù)的應(yīng)用程序尚未打開,此時(shí)要啟動(dòng)該應(yīng)用程序,然后轉(zhuǎn)入監(jiān)控流程。
/// <summary> /// 掃描進(jìn)程列表,判斷進(jìn)程對(duì)應(yīng)的全路徑是否與指定路徑一致 /// 如果一致,說明進(jìn)程已啟動(dòng) /// 如果不一致,說明進(jìn)程尚未啟動(dòng) /// </summary> /// <param name="strAddress"></param> private void ScanProcessList(string address) { Process[] arrayProcess = Process.GetProcesses(); foreach (Process p in arrayProcess) { //System、Idle進(jìn)程會(huì)拒絕訪問其全路徑 if (p.ProcessName != "System" && p.ProcessName != "Idle") { try { if (this.FormatPath(address) == this.FormatPath(p.MainModule.FileName.ToString())) { //進(jìn)程已啟動(dòng) this.WatchProcess(p, address); return; } } catch { //拒絕訪問進(jìn)程的全路徑 this.SaveLog("進(jìn)程(" + p.Id.ToString() + ")(" + p.ProcessName.ToString() + ")拒絕訪問全路徑!"); } } } //進(jìn)程尚未啟動(dòng) Process process = new Process(); process.StartInfo.FileName = address; process.Start(); this.WatchProcess(process, address); }
分配一條線程,執(zhí)行監(jiān)控任務(wù):
/// <summary> /// 監(jiān)聽進(jìn)程 /// </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> /// 構(gòu)造函數(shù) /// </summary> public ProcessRestart() {} /// <summary> /// 構(gòu)造函數(shù) /// </summary> /// <param name="process"></param> /// <param name="address"></param> public ProcessRestart(Process process, string address) { this._process = process; this._address = address; } /// <summary> /// 重啟進(jìn)程 /// </summary> public void RestartProcess() { try { while (true) { this._process.WaitForExit(); this._process.Close(); //釋放已退出進(jìn)程的句柄 this._process.StartInfo.FileName = this._address; this._process.Start(); Thread.Sleep(1000); } } catch (Exception ex) { ProcessWatcher objProcessWatcher = new ProcessWatcher(); objProcessWatcher.SaveLog("RestartProcess() 出錯(cuò),監(jiān)控程序已取消對(duì)進(jìn)程(" + this._process.Id.ToString() +")(" + this._process.ProcessName.ToString() + ")的監(jiān)控,錯(cuò)誤描述為:" + ex.Message.ToString()); } } }
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
C#中OpenCVSharp實(shí)現(xiàn)輪廓檢測(cè)
這篇文章主要介紹了C#中OpenCVSharp實(shí)現(xiàn)輪廓檢測(cè),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11C#實(shí)現(xiàn)Bitmap類型與Byte[]類型相互轉(zhuǎn)化的示例詳解
在C#編程中,Bitmap類型和Byte[]類型之間的相互轉(zhuǎn)化是圖像處理和數(shù)據(jù)傳輸中常見的需求,Bitmap類型表示一個(gè)位圖圖像,而Byte[]類型則是一個(gè)字節(jié)數(shù)組,本文將詳細(xì)介紹如何在這兩種類型之間進(jìn)行相互轉(zhuǎn)化,需要的朋友可以參考下2024-07-07Unity實(shí)現(xiàn)動(dòng)物識(shí)別的示例代碼
本文主要介紹了如何通過Unity實(shí)現(xiàn)動(dòng)物識(shí)別,可以實(shí)現(xiàn)識(shí)別近八千種動(dòng)物,接口返回動(dòng)物名稱,并可獲取識(shí)別結(jié)果對(duì)應(yīng)的百科信息,感興趣的可以了解一下2022-02-02DirectoryInfo引用一個(gè)相對(duì)目錄的實(shí)例
這種特殊參數(shù)在Windows的命令提示符或者“運(yùn)行”對(duì)話框中都可以使用,等價(jià)于DOS中的cd命令參數(shù)。直接上代碼,一看你就懂了:2013-04-04