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

转载于:https://www.cnblogs.com/dekevin/p/4096706.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
    前段时间有点空闲就自学了一下c#,以前也没有学过任何编程,只是对电脑操作还是很熟,会一些应用类软件,计算机相关的英文能看懂点。     看了10多节的视频教程,因为基础太差,前面学的东西老会忘,光这样学也没有多大的意义,于是就找个程序一边练一遍学。网上一搜,不知道怎么的就找了一个进程守护的软件,别人提供的源码,好像是在pudn下的,一测试可以运行,就是觉得不太适合自己的操作习惯。于是就开始改造了。牺牲了一周的业余时间,终于完成,在这分享一下,如果有需要的可以自行再改造,版权归原作者,至于我修改的地方,可以任意改,任意传播,没有版权。附上源码,包含源程序的和我改造的部分。只是希望能给像我这样的不会编程的一类朋友一点启示,只要努力,就能达到自己想要的效果。现成的东西,不一定是适合自己的,要想拿来适合自己,也许就需要付出努力。附上修改的说明及截图。没有多大的技术含量,对于会编程的人,也许2下就搞定了,对于不会编程的人,如果刚好需要这样的软件,那就可以偷懒了。测试了win7(x86x64),win8(x86),win10(x64),其他系统没有测试,如果不能运行请自行解决,vs2010编译的,dedug和Release都有编译好的版本。     修改及使用说明 1、配置文件precesses.xml,里面的进程名不要有扩展名,一定要注意。 2、配置文件中的mutl为是否多开,linecount为多开时的限制数,jgtime为守护的周期。 3、配置文件中新增linecount为相同程序限制数(如果程序本身不允许多开那就没办法),可以给每个程序配置不同限制数,但是界面上修改时,只能修改成一样。读取配置后显示最大的那个值在界面上. 4、配置文件中新增jgtime为循环时间,所有程序使用相同的循环时间,配置时可以配置成不一样的,实际没有意义,因只调用了一个定时器,但默认显示并使用最大的那个时间。 5、增加托盘图标提示及菜单 6、取消任务栏图标显示 7、默认启动最小化,如果没有任务栏,只能通过进程结束,固定了窗口不能改变大小 8、主界面增加了退出按钮、清空日志按钮 9、主界面点击关闭按钮后,自动最小化 10、增加进程调用的部分日志显示框,主要是调试方便,日志超过30行自动清屏,未保存,可修改代码关掉或不显示,evertlog目录下多余的日志关掉了,仍然有小部分日志,,如有必要自行在代码中关掉即可。 11、底部增加多开线程输入框及保存按钮,多开进程数大于1小于100,100为无限制模式。增加间隔时间输入框及保存按钮,间隔时间大于等于500,小于等于1000000000之间的整数(毫秒),没有限制再配置中添加应用程序的个数。 12、开始监控及停止监控互锁显示。 13、默认优先读取配置中的最大的间隔时间,如果配置错误会自动初始化为23秒。 14、其他界面布局微调 15、代码自己随意修改太拖沓,会编程的可以进行大量优化。 16、没有做自守护,可以考虑用任务计划写个批处理,再守护它。win764位以上系统,防任务管理栏结束有难度哦。 17、我用的是系统自带的自定义shell启动它,还是很稳定,23秒是我测试的需要,自己可在代码中修改,我修改的地方还是加了不少注释的。 18、调用有些程序可能无法启动的情况,和本软件无关,可能是路径问题,有些程序不允许改变路径运行等,可自己在命令行下测试调用的软件是否运行,我遇到过类似的软件,加入环境变量也不行,只能在同一个目录下。 19、加入配置文件错误的一些纠缠机制(linecount没有会自动创建并赋值3,值为"",也会自动修复),当然错误的情况不够完善,只考虑了没有整项或者值为""的情况,其他情况没有处理,如果遇到其他情况只会导致程序停止,不能对配置的自修复而已,只针对linecout和jgtime做了处理,jgtime整项如果没有,会自动修复为23000毫秒。程序自修复配置文件后,会自动运行,如遇不能修复的问题根据catch出的问题,自行修改源码,加入到出错情况的处理。 20、只测试了win7x86x64,win8x86,win10x64,精简系统可能无法运行,自己测试了一个5年前的极度精简的win7系统,无法运行没有任何提示,也没有去验证到底是什么问题,但大多数精简系统都行。 21、增加鼠标悬停帮助提示,提示框可能有拖影的情况,自行修改源码中的时间参数,也可改用另一种呈现方式不会拖影,源码中有,就是没有那么好看,鼠标悬停提示有点耗cpu资源。 22、双击进程框列出的需要守护监控的进程,可以查看该进程更多详细信息。 23、配置文件中路径,一定要注意用\\表示,比如 d:\\bitboy\\bitboy.exe  还要有扩展名,name中的进程名不能有扩展名,否则找不到程序,不区分大小写。 24、本软件运行后,会自动启动配置程序路径下的软件,如果该软件不存在,则自动停止监控。所有多余的配置要自行删除,xml文件中不要有注释,没对注释做处理。 25、加入开机自启动选择项,保存到注册表中,加入了默认以管理员身份运行,解决配置文件不在system32下,开机启动找不到配置文件的问题。 26、没有对本程序做单一启动的限制,如果需要只运行一个实例,重新编译主程序中注释掉的一部分代码。对多线程运行的程序,建议设置linecount为100,否则会杀死多余的线程,一个循环周期杀一个,线程数量固定不变的可在配置中设置相同的linecount数,这样能达到守护的目的。 27、加了开机启动参数的支持,在配置文件中路径后用英文的逗号作分隔,前面的是路径,后面的是启动参数。 28、支持对多线程运行程序的退出守护,配置文件中,将linecount设置为100,并且要启用多开。当linecount配置为100后,不允许在界面上修改,只能手动打开配置文件修改。启动后检查进程如果没有运行,就运行一次,如果已经存在了则不处理,因为有些程序,要开启很多个线程运行,只有等所有进程都退出之后,才会启动一次守护的进程,之后又会交给程序自适应。 29、增加循环启动时间次数的累计显示,和所有程序累计启动次数不一样,这个指的是定时器执行的次数,不管是否启动了守护的程序,这个周期都是累加,可能没多大意义,另外那个是程序启动一次才加一次。 30、防止任务管理栏结束,甚至采用服务方式启动等,对于我来说太难了,服务启动又带ui交互,好像涉及到session0穿透的问题,能力有限,没有精力搞了。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值