基本信息
源码名称:C# 实现进程守护功能 通用于所有进程,非法关闭和关机依然适用 附完整源码下载
源码大小:0.30M
文件格式:.zip
开发语言:C#
更新时间:2013-07-19
友情提示:(无需注册或充值,赞助后即可获取资源下载链接)
嘿,亲!知识可是无价之宝呢,但咱这精心整理的资料也耗费了不少心血呀。小小地破费一下,绝对物超所值哦!如有下载和支付问题,请联系我们QQ(微信同号):78630559
本次赞助数额为: 2 元×
微信扫码支付:2 元
×
请留下您的邮箱,我们将在2小时内将文件发到您的邮箱
源码介绍
1 "%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe"
2 net start "CocoWatcher"
3 pause
如果你想卸载该守护程序,点击批处理文档“卸载.bat”,“卸载.bat”具体内容如下:
1 net stop "CocoWatcher"
2 "%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe" -u
3 taskkill /f /im CocoWatcher.exe
4 pause
3. 需求分析
用户指定要守护的应用程序(数量不限),该应用程序不仅包括exe可执行文件,还包括诸如jpg、txt等所有能双击打开执行的应用程序。用户设定好要守护的应用程序后,关闭应用程序(包括合法和非法关闭),该应用程序要能立即重启打开。当电脑重启时,要守护的应用程序也能自动全部打开。
4. 详细设计
要实现上述需求,首先要提供一个配置档,让用户能随意配置要守护的应用程序。那么,该配置档要配置应用程序的什么信息呢?答案:应用程序的全路径。
好,我们已经知道了要守护的应用程序的全路径,接下来怎样完成守护任务呢?首先,我们应该打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,此时要分配一条线程监控该进程句柄,当该进程句柄返回信息,说明该进程已关闭,此时释放进程句柄内存,并重启该进程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。
值得注意的是,一定要额外分配线程去监控要守护的应用程序,为什么?因为如果你用主线程(入口函数线程)去执行监控任务,会被长期阻塞,直到进程退出才会被激活,这样就无法运行后续程序。况且,监控程序要实现持续监控,要使用死循环,如果主线程进入死循环,就无法监控其他要守护的进程了。
1. 下载本实例源码
2. 安装注意事项
在配置档中配置你要守护的应用程序,应用程序之间用逗号隔开:
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所示:
1 "%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe"
2 net start "CocoWatcher"
3 pause
如果你想卸载该守护程序,点击批处理文档“卸载.bat”,“卸载.bat”具体内容如下:
1 net stop "CocoWatcher"
2 "%cd%\InstallUtil.exe" "%cd%\CocoWatcher.exe" -u
3 taskkill /f /im CocoWatcher.exe
4 pause
3. 需求分析
用户指定要守护的应用程序(数量不限),该应用程序不仅包括exe可执行文件,还包括诸如jpg、txt等所有能双击打开执行的应用程序。用户设定好要守护的应用程序后,关闭应用程序(包括合法和非法关闭),该应用程序要能立即重启打开。当电脑重启时,要守护的应用程序也能自动全部打开。
4. 详细设计
要实现上述需求,首先要提供一个配置档,让用户能随意配置要守护的应用程序。那么,该配置档要配置应用程序的什么信息呢?答案:应用程序的全路径。
好,我们已经知道了要守护的应用程序的全路径,接下来怎样完成守护任务呢?首先,我们应该打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,此时要分配一条线程监控该进程句柄,当该进程句柄返回信息,说明该进程已关闭,此时释放进程句柄内存,并重启该进程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。
值得注意的是,一定要额外分配线程去监控要守护的应用程序,为什么?因为如果你用主线程(入口函数线程)去执行监控任务,会被长期阻塞,直到进程退出才会被激活,这样就无法运行后续程序。况且,监控程序要实现持续监控,要使用死循环,如果主线程进入死循环,就无法监控其他要守护的进程了。
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Diagnostics;
using System.Linq;
using System.ServiceProcess;
using System.Text;
using System.IO;
using System.Configuration;
using System.Threading;
namespace CocoWatcher
{
public partial class ProcessWatcher : ServiceBase
{
//字段
private string[] _processAddress;
private object _lockerForLog = new object();
private string _logPath = string.Empty;
/// <summary>
/// 构造函数
/// </summary>
public ProcessWatcher()
{
InitializeComponent();
try
{
//读取监控进程全路径
string strProcessAddress = ConfigurationManager.AppSettings["ProcessAddress"].ToString();
if (strProcessAddress.Trim() != "")
{
this._processAddress = strProcessAddress.Split(',');
}
else
{
throw new Exception("读取配置档ProcessAddress失败,ProcessAddress为空!");
}
//创建日志目录
this._logPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "CocoWatcherLog");
if (!Directory.Exists(_logPath))
{
Directory.CreateDirectory(_logPath);
}
}
catch (Exception ex)
{
this.SaveLog("Watcher()初始化出错!错误描述为:" ex.Message.ToString());
}
}
/// <summary>
/// 启动服务
/// </summary>
/// <param name="args"></param>
protected override void OnStart(string[] args)
{
try
{
this.StartWatch();
}
catch (Exception ex)
{
this.SaveLog("OnStart() 出错,错误描述:" ex.Message.ToString());
}
}
/// <summary>
/// 停止服务
/// </summary>
protected override void OnStop()
{
try
{
}
catch (Exception ex)
{
this.SaveLog("OnStop 出错,错误描述:" ex.Message.ToString());
}
}
/// <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();
}
/// <summary>
/// 格式化路径
/// 去除前后空格
/// 去除最后的"\"
/// 字母全部转化为小写
/// </summary>
/// <param name="path"></param>
/// <returns></returns>
private string FormatPath(string path)
{
return path.ToLower().Trim().TrimEnd('\\');
}
/// <summary>
/// 记录日志
/// </summary>
/// <param name="content"></param>
public void SaveLog(string content)
{
try
{
lock (_lockerForLog)
{
FileStream fs;
fs = new FileStream(Path.Combine(this._logPath, DateTime.Now.ToString("yyyyMMdd") ".log"), FileMode.OpenOrCreate);
StreamWriter streamWriter = new StreamWriter(fs);
streamWriter.BaseStream.Seek(0, SeekOrigin.End);
streamWriter.WriteLine("[" DateTime.Now.ToString() "]:" content);
streamWriter.Flush();
streamWriter.Close();
fs.Close();
}
}
catch
{
}
}
}
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());
}
}
}
}