如何利用c#實現通用守護進程

這篇文章主要給大家介紹了關於如何利用c#實現通用守護進程的相關資料,文中通過示例代碼介紹的非常詳細,對大家學習或者使用c#具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧

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. 安裝注意事項

在配置檔中配置你要守護的應用程序,應用程序之間用逗號隔開:

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

總結

以上就是我在處理客戶端真實IP的方法,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對神馬文庫的支持。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章