如何利用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)用程序之間用逗號隔開:
<?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”會報(bào)錯,如圖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)電腦重啟時,要守護(hù)的應(yīng)用程序也能自動全部打開。
4. 詳細(xì)設(shè)計(jì)
要實(shí)現(xiàn)上述需求,首先要提供一個配置檔,讓用戶能隨意配置要守護(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)用程序的全路徑比對,如果一致,說明要守護(hù)的應(yīng)用程序已開啟了,此時要分配一條線程監(jiān)控該進(jìn)程句柄,當(dāng)該進(jìn)程句柄返回信息,說明該進(jìn)程已關(guān)閉,此時釋放進(jìn)程句柄內(nèi)存,并重啟該進(jìn)程。如果遍歷任務(wù)管理進(jìn)程列表中所有進(jìn)程,沒有找到與要守護(hù)的應(yīng)用程序的全路徑一致的進(jìn)程,說明要守護(hù)的應(yīng)用程序尚未打開,此時要啟動該應(yīng)用程序,然后轉(zhuǎn)入監(jiān)控流程。
值得注意的是,一定要額外分配線程去監(jiān)控要守護(hù)的應(yīng)用程序,為什么?因?yàn)槿绻阌弥骶€程(入口函數(shù)線程)去執(zhí)行監(jiān)控任務(wù),會被長期阻塞,直到進(jìn)程退出才會被激活,這樣就無法運(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ā)步驟,請參考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)用程序的全路徑比對,如果一致,說明要守護(hù)的應(yīng)用程序已開啟了,進(jìn)入監(jiān)控流程。如果遍歷任務(wù)管理進(jìn)程列表中所有進(jìn)程,沒有找到與要守護(hù)的應(yīng)用程序的全路徑一致的進(jìn)程,說明要守護(hù)的應(yīng)用程序尚未打開,此時要啟動該應(yīng)用程序,然后轉(zhuǎn)入監(jiān)控流程。
/// <summary>
/// 掃描進(jìn)程列表,判斷進(jìn)程對應(yīng)的全路徑是否與指定路徑一致
/// 如果一致,說明進(jìn)程已啟動
/// 如果不一致,說明進(jìn)程尚未啟動
/// </summary>
/// <param name="strAddress"></param>
private void ScanProcessList(string address)
{
Process[] arrayProcess = Process.GetProcesses();
foreach (Process p in arrayProcess)
{
//System、Idle進(jìn)程會拒絕訪問其全路徑
if (p.ProcessName != "System" && p.ProcessName != "Idle")
{
try
{
if (this.FormatPath(address) == this.FormatPath(p.MainModule.FileName.ToString()))
{
//進(jìn)程已啟動
this.WatchProcess(p, address);
return;
}
}
catch
{
//拒絕訪問進(jìn)程的全路徑
this.SaveLog("進(jìn)程(" + p.Id.ToString() + ")(" + p.ProcessName.ToString() + ")拒絕訪問全路徑!");
}
}
}
//進(jìn)程尚未啟動
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() 出錯,監(jiān)控程序已取消對進(jìn)程("
+ this._process.Id.ToString() +")(" + this._process.ProcessName.ToString()
+ ")的監(jiān)控,錯誤描述為:" + ex.Message.ToString());
}
}
}
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
C#中OpenCVSharp實(shí)現(xiàn)輪廓檢測
這篇文章主要介紹了C#中OpenCVSharp實(shí)現(xiàn)輪廓檢測,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11
C#實(shí)現(xiàn)Bitmap類型與Byte[]類型相互轉(zhuǎn)化的示例詳解
在C#編程中,Bitmap類型和Byte[]類型之間的相互轉(zhuǎn)化是圖像處理和數(shù)據(jù)傳輸中常見的需求,Bitmap類型表示一個位圖圖像,而Byte[]類型則是一個字節(jié)數(shù)組,本文將詳細(xì)介紹如何在這兩種類型之間進(jìn)行相互轉(zhuǎn)化,需要的朋友可以參考下2024-07-07

