通用守护进程的实现
2011-02-12 11:55 田志良 阅读(23799) 评论(42) 编辑 收藏 举报1. 下载
源码下载:https://files.cnblogs.com/tianzhiliang/CocoWatcher.rar
安装包下载:https://files.cnblogs.com/tianzhiliang/CocoWatcher_Setup.rar
2. 安装注意事项
- 在配置档中配置你要守护的应用程序,应用程序之间用逗号隔开:
12345678910
<?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”具体内容如下:
123"%cd%\InstallUtil.exe"
"%cd%\CocoWatcher.exe"
net start
"CocoWatcher"
pause
1234net stop
"CocoWatcher"
"%cd%\InstallUtil.exe"
"%cd%\CocoWatcher.exe"
-u
taskkill /f /im CocoWatcher.exe
pause
3. 需求分析
用户指定要守护的应用程序(数量不限),该应用程序不仅包括exe可执行文件,还包括诸如jpg、txt等所有能双击打开执行的应用程序。用户设定好要守护的应用程序后,关闭应用程序(包括合法和非法关闭),该应用程序要能立即重启打开。当电脑重启时,要守护的应用程序也能自动全部打开。
4. 详细设计
要实现上述需求,首先要提供一个配置档,让用户能随意配置要守护的应用程序。那么,该配置档要配置应用程序的什么信息呢?答案:应用程序的全路径。
好,我们已经知道了要守护的应用程序的全路径,接下来怎样完成守护任务呢?首先,我们应该打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,此时要分配一条线程监控该进程句柄,当该进程句柄返回信息,说明该进程已关闭,此时释放进程句柄内存,并重启该进程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。
值得注意的是,一定要额外分配线程去监控要守护的应用程序,为什么?因为如果你用主线程(入口函数线程)去执行监控任务,会被长期阻塞,直到进程退出才会被激活,这样就无法运行后续程序。况且,监控程序要实现持续监控,要使用死循环,如果主线程进入死循环,就无法监控其他要守护的进程了。
5. 代码详解
Windows服务的开发步骤,请参考MSDN,此处略去。下面将关键代码贴出,加以解释。
读取配置档中“ProcessAddress”节点,获取要守护的应用程序全目录,验证应用程序全目录,如果合法,进入扫描任务管理器进程列表流程。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 | /// <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()); } } } } } } |
打开任务管理器,查看一下正在运行的有哪些进程,然后逐一读取出这些进程的全路径,与要守护的应用程序的全路径比对,如果一致,说明要守护的应用程序已开启了,进入监控流程。如果遍历任务管理进程列表中所有进程,没有找到与要守护的应用程序的全路径一致的进程,说明要守护的应用程序尚未打开,此时要启动该应用程序,然后转入监控流程。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 | /// <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); } |
分配一条线程,执行监控任务:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 | /// <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()); } } } |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 三行代码完成国际化适配,妙~啊~
· .NET Core 中如何实现缓存的预热?