用c#实现通用守护进程

    技术2025-06-02  88

    http://www.cnblogs.com/tianzhiliang/archive/2011/02/12/1952221.html

     

    1. 下载

      源码下载:http://files.cnblogs.com/tianzhiliang/CocoWatcher.rar

      安装包下载:http://files.cnblogs.com/tianzhiliang/CocoWatcher_Setup.rar

    2. 安装注意事项

    在配置档中配置你要守护的应用程序,应用程序之间用逗号隔开: view source print ? 01<?xml version="1.0" encoding="utf-8" ?> 02<configuration> 03  <appSettings> 04    <add key="ProcessAddress" value=" 05         d:/war3.exe, 06         d:/note.txt, 07         d:/girl.jpg 08    " /> 09  </appSettings> 10</configuration> 该项目是Windows服务,直接打开“CocoWatcher.exe”会报错,如图1所示:运行该守护程序需安装Windows服务,点击批处理文档“安装.bat”即可,“安装.bat”具体内容如下: view source print ? 1"%cd%/InstallUtil.exe" "%cd%/CocoWatcher.exe"  2net start "CocoWatcher"  3pause 如果你想卸载该守护程序,点击批处理文档“卸载.bat”,“卸载.bat”具体内容如下: view source print ? 1net stop "CocoWatcher"  2"%cd%/InstallUtil.exe" "%cd%/CocoWatcher.exe"  -u 3taskkill /f /im CocoWatcher.exe 4pause

    3. 需求分析

      用户指定要守护的应用程序(数量不限),该应用程序不仅包括exe可执行文件,还包括诸如jpg、txt等所有能双击打开执行的应用程序。用户设定好要守护的应用程序后,关闭应用程序(包括合法和非法关闭),该应用程序要能立即重启打开。当电脑重启时,要守护的应用程序也能自动全部打开。

    4. 详细设计

      要实现上述需求,首先要提供一个配置档,让用户能随意配置要守护的应用程序。那么,该配置档要配置应用程序的什么信息呢?答案:应用程序的全路径。

      好,我们已经知道了要守护的应用程序的全路径,接下来怎样完成守护任务呢?首先,我们应该打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,此时要分配一条线程监控该进程句柄,当该进程句柄返回信息,说明该进程已关闭,此时释放进程句柄内存,并重启该进程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。

      值得注意的是,一定要额外分配线程去监控要守护的应用程序,为什么?因为如果你用主线程(入口函数线程)去执行监控任务,会被长期阻塞,直到进程退出才会被激活,这样就无法运行后续程序。况且,监控程序要实现持续监控,要使用死循环,如果主线程进入死循环,就无法监控其他要守护的进程了。

    5. 代码详解

      Windows服务的开发步骤,请参考MSDN,此处略去。下面将关键代码贴出,加以解释。

      读取配置档中“ProcessAddress”节点,获取要守护的应用程序全目录,验证应用程序全目录,如果合法,进入扫描任务管理器进程列表流程。

    view source print ? 01/// <summary> 02/// 开始监控 03/// </summary> 04private void StartWatch() 05{ 06    if (this._processAddress != null) 07    { 08        if (this._processAddress.Length > 0) 09        { 10            foreach (string str in _processAddress) 11            { 12                if (str.Trim() != "") 13                { 14                    if (File.Exists(str.Trim())) 15                    { 16                        this.ScanProcessList(str.Trim()); 17                    } 18                } 19            } 20        } 21    } 22}

      打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,进入监控流程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。

    view source print ? 01/// <summary> 02/// 扫描进程列表,判断进程对应的全路径是否与指定路径一致 03/// 如果一致,说明进程已启动 04/// 如果不一致,说明进程尚未启动 05/// </summary> 06/// <param name="strAddress"></param> 07private void ScanProcessList(string address) 08{ 09    Process[] arrayProcess = Process.GetProcesses(); 10    foreach (Process p in arrayProcess) 11    { 12        //System、Idle进程会拒绝访问其全路径 13        if (p.ProcessName != "System" && p.ProcessName != "Idle") 14        { 15            try 16            { 17                if (this.FormatPath(address) == this.FormatPath(p.MainModule.FileName.ToString())) 18                { 19                    //进程已启动 20                    this.WatchProcess(p, address); 21                    return; 22                } 23            } 24            catch 25            { 26                //拒绝访问进程的全路径 27                this.SaveLog("进程(" + p.Id.ToString() + ")(" + p.ProcessName.ToString() + ")拒绝访问全路径!"); 28            } 29        } 30    } 31       32    //进程尚未启动 33    Process process = new Process(); 34    process.StartInfo.FileName = address; 35    process.Start(); 36    this.WatchProcess(process, address); 37}

      分配一条线程,执行监控任务:

    view source print ? 01    /// <summary> 02    /// 监听进程 03    /// </summary> 04    /// <param name="p"></param> 05    /// <param name="address"></param> 06    private void WatchProcess(Process process, string address) 07    { 08        ProcessRestart objProcessRestart = new ProcessRestart(process, address); 09        Thread thread = new Thread(new ThreadStart(objProcessRestart.RestartProcess)); 10        thread.Start(); 11    } 12  13  14public class ProcessRestart 15{ 16    //字段 17    private Process _process; 18    private string _address; 19  20  21    /// <summary> 22    /// 构造函数 23    /// </summary> 24    public ProcessRestart() 25    {} 26  27  28    /// <summary> 29    /// 构造函数 30    /// </summary> 31    /// <param name="process"></param> 32    /// <param name="address"></param> 33    public ProcessRestart(Process process, string address) 34    { 35        this._process = process; 36        this._address = address; 37    } 38  39  40    /// <summary> 41    /// 重启进程 42    /// </summary> 43    public void RestartProcess() 44    { 45        try 46        { 47            while (true) 48            { 49                this._process.WaitForExit(); 50                this._process.Close();    //释放已退出进程的句柄 51                this._process.StartInfo.FileName = this._address; 52                this._process.Start(); 53  54                Thread.Sleep(1000); 55            } 56        } 57        catch (Exception ex) 58        { 59            ProcessWatcher objProcessWatcher = new ProcessWatcher(); 60            objProcessWatcher.SaveLog("RestartProcess() 出错,监控程序已取消对进程(" 61                + this._process.Id.ToString() +")(" + this._process.ProcessName.ToString()  62                + ")的监控,错误描述为:" + ex.Message.ToString()); 63        } 64    } 65}
    最新回复(0)