用C#實現通用守護進程


1. 下載

  源碼下載:http://pan.baidu.com/s/1vqDA2

  安裝包下載:http://pan.baidu.com/s/1sjmEB0p

2. 安裝注意事項

  • 在配置檔中配置你要守護的應用程序,應用程序之間用逗號隔開
<?xml version="1.0" encoding="utf-8" ?>
<configuration>
  <appSettings>
    <add key="ProcessAddress" value="
         d:\war3.exe,
         d:\note.exe,
         d:\girl.jpg
    " />
  </appSettings>
</configuration>

 該項目是Windows服務,直接打開“CocoWatcher.exe”會報錯,如圖1所示:

這個時候先運行【installutil.exe】文件,然后運行該守護程序需安裝Windows服務,點擊批處理文檔【安裝.bat】即可,運行之后進入到【管理工具】-【服務】中就能看到有CocoWatcher服務已經啟動了,這個時候配置文件中的進程就被保護起來了。這個時候再去任務管理器中去關閉配置文件中的進程就可以發現,已經不能關閉了。

【安裝.bat】具體內容如下:

"%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe"
net start "CocoWatcher"
pause

如果你不想守護進程了,就需要卸載該守護程序,點擊批處理文檔【卸載.bat】,【卸載.bat】文件具體內容如下:

net stop "CocoWatcher"
"%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe"  -u
taskkill /f /im CocoWatcher.exe
pause

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());
        }
    }
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM