用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());
        }
    }
}
时间: 2024-08-01 11:08:21

用C#实现通用守护进程的相关文章

[C#]通用守护进程服务

摘要 很多情况下,都会使用windows服务做一些任务,但总会有一些异常,导致服务停止.这个时候,开发人员又不能立马解决问题,所以做一个守护者服务还是很有必要的.当检测到服务停止了,重启一下服务,等开发人员到位了,再排查错误日志. 代码 app.config <?xml version="1.0" encoding="utf-8" ?> <configuration> <startup> <supportedRuntime

Linux系统开发7 进程关系,守护进程

[本文谢绝转载原文来自http://990487026.blog.51cto.com] <大纲> Linux系统开发7  进程关系守护进程 终端 网络终端 Linux PCB结构体信息 进程组 修改子进程.父进程的组ID 会话组 设置一个会话脱离控制终端 生成一个新的会话 守护进程 守护进程模板 获取当前系统时间  终端 在UNIX系统中用户通过终端登录系统后得到一个Shell进程这个终端成为Shell进 程的控制终端Controlling Terminal在讲进程时讲过控制终端是保存在PCB

linux守护进程解读

Linux系统守护进程详解 不要关闭下面这几个服务: acpid, haldaemon, messagebus, klogd, network, syslogd 1. NetworkManager, NetworkManagerDispatcher NetworkManager 是一个自动切换网络连接的后台进程.很多笔记本用户都需要启用该功能,它让你能够在无线网络和有线网络之间切换.大多数台式机用户应该关闭该服务.一些 DHCP 用户可能需要开启它. 2. acpid 配置文件:/proc/ac

数组,父子进程,守护进程

2015.1.25星期日,阴天 二维数组a中共有m行n列个元素从a[0][0]到a[i][j]之间共有i*n+j个元素p代表第0行第0列的地址,所以元素a[i][j]的地址为 p + i*n+j a[i][j] == p[i*n+j] == *(p + i*n+j) 在Linux中获得当前进程的PID和PPID的系统调用函数为getpid()和getppid();获得当前PID和PPID后可以将其写入日志备份! #include <stdio.h>#include <unistd.h&g

Linux学习之守护进程详解

Linux系统守护进程详解                                                              ---转自:http://yuanbin.blog.51cto.com/363003/107306/,多谢分享 不要关闭下面这几个服务: acpid, haldaemon, messagebus, klogd, network, syslogd 1. NetworkManager, NetworkManagerDispatcher NetworkM

《Unix环境高级编程》读书笔记 第13章-守护进程

1. 引言 守护进程是生存期长的一种进程.它们常常在系统引导装入时启动,仅在系统关闭时才终止.它们没有控制终端,在后台运行. 本章说明守护进程结构.如何编写守护进程程序.守护进程如何报告出错情况. 2. 守护进程的特征 基于BSD的系统下执行:ps -axj -a 显示由其他用户所拥有的进程的状态:-x 显示没有控制终端的进程状态:-j 显示与作业有关的信息 基于System V的系统下执行:ps -efj Linux下执行以上两个命令输出一致 常见的守护进程: kswapd,内存换页守护进程.

Linux--终端、作业控制、与守护进程

一.进程组,作业,会话的概念 1.进程组:是一个或多个进程的集合.通常,与同一作业相关联,可以接收来自同一终端的各种信号.每个进程都有一个唯一的进程组ID.每个进程组都可以有一个组长进程.组长进程的标识是,其进程组ID等于其进程ID.组长进程可以创建一个进程组,创建该组中的进程,然后终止.只要在某个进程组中有一个进程存在,则该进程组就存在,这与组长进程是否终止无关. 2.作业:Shell分前后台来控制的不是进程而是作业(job)或者进程组.一个前台作业可以由多个进程组成,一个后台也可以由多个进程

ansible 部署基于windows 2008 r2+rsync+alwaysUp(守护进程)

准备:升级PowerShell3.0 及打开WinRM模块 参考文章:http://juestnow.blog.51cto.com/1515305/1875026 下载软件包rsync http://down.51cto.com/data/371729 alwaysUp https://www.coretechnologies.com/products/AlwaysUp/ AlwaysUpCLT https://www.coretechnologies.com/products/AlwaysUp

Linux多任务编程之七:Linux守护进程及其基础实验(转)

来源:CSDN  作者:王文松  转自Linux公社 ------------------------------------------------------------------------------------------------ 守护进程概述 守护进程,又叫daemon进程(不知怎的,我突然想起来吸血鬼日记中的达蒙了,很好看的美剧),是Linux中的后台服务进程. 他是一个生存期较长的进程,通常独立于控制终端并且周期性地执行某种任务或者等待处理某些发生的事件.守护进程常常在系统