C++ 监控服务和应用程序

用C++写一个监控服务和应用程序状态的看门狗程序

由于公司业务需要,写了一个看门狗程序。

主要功能

1.隔一小段时间检测一下服务或应用程序的状态。
2.如果服务或应用程序异常停止或者关闭,将其重新启动

上代码上代码

#include<iostream>
#include<Windows.h>
#include<shlwapi.h>
#include<atltime.h>
#include <TLHELP32.H>     // 声明快照函数的头文件

HWND GetWindowHandleByPID(DWORD dwProcessID)
{
	HWND h = GetTopWindow(0);
	while (h)
	{
		DWORD pid = 0;
		DWORD dwTheardId = GetWindowThreadProcessId(h, &pid);

		if (dwTheardId != 0)
		{
			if (pid == dwProcessID/*your process id*/)
			{
				// here h is the handle to the window
				return h;
			}
		}

		h = GetNextWindow(h, GW_HWNDNEXT);
	}

	return NULL;
}


void OnTimer(UINT_PTR nIDEvent)
{
	while (true)
	{
		BOOL bFind = FALSE;
		PROCESSENTRY32 pe32;
		pe32.dwSize = sizeof(pe32);

		// 给系统内的所有进程拍一个快照--改函数用于获取系统指定进程的快照,也可以传入不同参数获取被这些进程使用的堆、模块和线程的快照
		HANDLE hProcessSnap = ::CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
		if (hProcessSnap == INVALID_HANDLE_VALUE)
		{
			//printf("CreateToolhelp32Snapshot 调用失败!\n");
			//logger.LogInfo("CreateToolhelp32Snapshot 调用失败!");
		}

		// 遍历进程快照,轮流显示每个进程的信息
		BOOL bMore = ::Process32First(hProcessSnap, &pe32);
		while (bMore)
		{
			if (!strcmp(pe32.szExeFile, "被监控的进程名"))//服务运行的时候也有一个进程名字
			{
				bFind = TRUE;
				//logger.LogInfo("Found it");
			}
			bMore = ::Process32Next(hProcessSnap, &pe32);
		}

		// 不要忘记清除掉snapshot 对象
		::CloseHandle(hProcessSnap);
		if (!bFind)          //如果没有VServer启动
		{
			//如果不需要监控服务的话,可以把下面这段代码注释掉。
			DWORD err;
			SC_HANDLE sc_handle = NULL;
			sc_handle = OpenSCManagerA(NULL, NULL, SERVICE_START);
			if (sc_handle == NULL)
			{
				err = GetLastError();
				//logger.LogInfo("OpenSCManagerA failed,Error:%d", err);
			}
			sc_handle = OpenServiceA(sc_handle, "被监控的服务名", SERVICE_START);
			if (sc_handle == NULL)
			{
			    err = GetLastError();
				//logger.LogInfo("Open server failed,Error:%d", err);
			}
			if (!StartServiceA(sc_handle, NULL, NULL))
			{
				err = GetLastError();
				//logger.LogInfo("Start server failed,Error:%d", err);
			}
			else
				//logger.LogInfo("Start the server success");

			if (sc_handle != NULL)
			{
				CloseServiceHandle(sc_handle);
			}
			
			//上面的代码是监控服务用的。下面的代码是监控应用程序用的,需要就打开
			//WinExec(strAppPath, SW_NORMAL);
		}
		else
		{
			FILETIME CreationTime = { 0 };
			FILETIME LastAccessTime = { 0 };
			FILETIME LastWriteTime = { 0 };
			FILETIME LocalTime = { 0 };

			SYSTEMTIME sysTime;
			//被监控的程序需要开一个线程 隔几秒钟打开关闭一下check.txt文件
			HANDLE hFile = CreateFile(".//check.txt", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, NULL, NULL);
			if (INVALID_HANDLE_VALUE != hFile)
			{
				GetFileTime(hFile, &CreationTime, &LastAccessTime, &LastWriteTime);

				FileTimeToLocalFileTime(&LastWriteTime, &LocalTime);
				FileTimeToSystemTime(&LocalTime, &sysTime);

				CTime time1 = CTime::GetCurrentTime();
				CTime time2((int)sysTime.wYear, (int)sysTime.wMonth, (int)sysTime.wDay,
					(int)sysTime.wHour, (int)sysTime.wMinute, (int)sysTime.wSecond);
				CTimeSpan ts = time1 - time2;

				if (ts.GetTotalSeconds() >= 30)   //大于90秒没响应就关闭程序
				{
					//logger.LogInfo("demo.exe 90s no response,let me shut it down");
					HWND hWnd = GetWindowHandleByPID(pe32.th32ProcessID);
					if (NULL != hWnd)
					{
						HANDLE hprocess = ::OpenProcess(PROCESS_ALL_ACCESS, FALSE, pe32.th32ProcessID);
						TerminateProcess(hprocess, 0);
					}
				}
				CloseHandle(hFile);
			}
		}
		//每隔15秒检测一次
		for (int i = 0; i < 15; ++i)
			Sleep(1000);
	}
}

int main()
{

	UINT_PTR nIDEvent;
	OnTimer(nIDEvent);
	getchar();
	return 0;
}

以上就是程序的全部代码,如果需要将其做成服务的话可以参考这篇文章:https://blog.csdn.net/weixin_44229493/article/details/109527911

  • 1
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
主要功能: 1.读取配置文件程序 2.启动进程 3.监控进程,查看进程是否退出或者崩溃 4.若进程退出或者崩溃,重启程序。 5.支持sleep功能 6.进程若连续崩溃NUM_MAX次就进行相应的睡眠周期struct proc_struct proc: struct proc_ struct [mp: if(array) return 0 ∥切换到目录rse chdirldiri ifdp= opendir(dir}=NuLL}开日录/proc,矢败返回0,成功把描述指针返回给d return o 〃将φpro文件夹的描述符指针传递给reεddir,读取文件夹内容,循环赋值给结构体di while ((dirp= readdir(dp))= NULLY char data 301 ∥取文件名称赋值给数组daa(其中包含有进程的名称(pid sprintf(data, "s", dirp->d_name); ∥是否是由字符09组成的字符串,即得到所有进程的pid f((IsDigit(data)) prac =(struct proc_struct )4 malloc(sizeof(struct proc_struct) tmp proc: prac->pid =a: oi(dirp->d_name): It(proc tind( proc. array)) free( tmp); closedir(dp cturn proc_find 两个参数分别是两个进程描述的结构体指针 李比较两个进程pd是否相等 李*相等返回1,不相等返回0 幸率球事容球家草事家事球峰率享事球摩率球享享溶事*事卷寒球套事塞容寒/ int proc find( struct prcc_struct* src, struct proc- struct* dest) char buffer[40%6]. ps cmd[20] It fd. I sprintf(buffer, "ed/star", sre->pid); fd = open(butter, O_RDONLY) if(fd==-1) rerurn 0 memset(buffer, wO, sizeof(buffer)) len= read(fd, bufter, sizeof(bufter )-1) close(ld) if(l return 0: p= butter: p= strrchr(p, C) narq=strrchr(p, )) n=q-p-1 if (len >= sizeof, srt->name)) len= sizeof(src->name)-1 p+ l, len src->namelen]=0; =日 turn(strcmp( src->name, dest dest->name)==0)? 1: 0- 条善参数aay:让程结构体指针;参数sie进程列表数组aray的大小ie:配置文件路径 从配置文件得到指定的程序列表,将对应进程的信息填充到aray数组中 羋执行成功返回进程个数,执行失败返回0 int get_ proc( struct proc_struct array, int size, char file intnRet=o if(! array I‖(si 0)l‖fhle myprinttf"invalid parameterin retun o char line[4096]; FILE fp= fopen(file, T"); if(fp) printf("open file cs fail\n", file) return U memset(line, 0, 4095); while(fgets(lire, 4095, tp)&& nRet size) memcpy(void s)[(&arraylnRet )->cmdline), (void")line, strlen(line)-2 ) tmp= strrchr(line, / ) Lmp += I: memcpy((&array inRet))->name, tmp, strlen(tmp)- 2) nRet++ ); return(nReL); 康棒串串浓凉率旅浓串底率卖毒志着旅浓浓准溶房表 装 startProc *卷参数proc:要启动的进的结构体描述指针 启动程序 执行成功返回1,子进程退出 宗塞家康家家家家家家家家宋家家聚家苯家球察塞家塞家家容家塞家家家家室家家察家家家聚家聚寒撑家装家掌建察家家室事 int startProc (struct proc_ struct* proc, struct proc _struct*must_run_ proc int mIst_run_size static inti=d if( proc)return 0 if(strlen(proc->cmdline I<=0) return 0; int pid= forko: 〃进程内部返回值为0,返回给父进程的为自己的pid inta〓 if(pid pid= fork( ifpd≡0 execl(char")proc->cmdline,(char")prDc->name,NULL); ∥exit: It(o): sleep42片 waiL(NULL) sleep( I: if(i== must run size -1) if(check proc(&must run proc[i])==0) startProc( &mtust_run_proeli], must_run_prce, must_run_size); el if(i== must run size-11 i= else 1++ start Proc( &must_run_proclil, must_run_ proc, mustrun_ _size); !**幸幸串率幸米幸*家*幸毕零*幸幸半字幸字华米*幸半孝率非幸零幸学幸幸车 3a*8*daemon init 幸*启动配置文件当中的需要守护的程序 执行成功返回1,中途出错,返回-1 长界零家墨军零家零率家三哮零座零率零零容岸军零罕型率零零零零牢察座察零零零零季球军零容零 int moniter_ run(struct proc_struct"must_run_proc, int proc_ size) nti=0: for(i=0; i< must_run_size: i ++) ∥监控程序是否正在运行 if(check_ proc(&(must un_ proc[il))<=o) ∥厘新片动程序 startProc(&' must run procli]), must run proc, proc size return I: 幸*事率事率率**率**字幸学摩*率*幸幸学幸半*率幸字****幸中*幸学幸 春*着*信号处理函数 exit_proc 翥安全结束监控的程序 4来没有返回值 告参毒萨响幸帝称昨嗜幸古称索点响卷南都南请南幸难布际本啪昨青市南动南香请非市赤南本 void exit_ proc(int ar InL I struct proc struct proc for(i=0; i< must run_ Size: i++) proc=&(must_run_proc[i]): kill(proc->pid, SIGTERM); exit flag=I exit(o): void exit_proc(int pid 要main L.获取程序列表2启动进程3.监控进程,若程序岀或崩溃,重新启动程序4.收到退 出信号,安全结束监控程序 成功返回1,失败返回0 零牢容容家容字家容容察*禁容容字哮零常字容容容家察容牢容零容容容容容牢字家客字容牢容零容*字容客字容容字容家容容字岩 static void run moniter( void data) 读取程序列表 must_ run _size get proc(must_run_proc, sIZE, data if(rmust run Sizc <=1) return o struct sigaction act, oldact act,sa handler= exit_proc act. sa flags =SA ONESHOT sigaction(SIGTERM, &act, NULL) sigaction(SIGINT, &act, NULL) sigaction(sIGHUP, &act, NULL); 检测并启动未启动的程序 moniter_ run(must run proc, must run slze) eturn null int creat and run moniter(char * file) 开线程: pthread_t moniter_ thread if(pthread_create(&moniter_thread, NULL, run_moniter, file)==0) printf("thread create Ok, check thread start \n") return printf( thread check create Errin"): return -I 零零零零享享事职增零半非寥零享半容零摩率率零享剩率容半半享零半率零半率零率辱寒零享 要 IsDigit 参茶爹数a字符串的地址 *判断字符串是否有纯数字字符组成 春客是纯数字返回1,不是纯数字返回0 喜非要串思率串串串串家串润串串串串串串毒毒喜串串最率毒串串踪串率串串非球毒串妆串串毒串串影零串串毒事串 static int IsDigit[char aD) Int size ∥得到当前字符串的长度 size= strlen(a: ∥若字符串长度为0,则直接返回0:即宇符串为空则返回0: if(size ==0) return 0; ∥循环遍历整个字符串 forli=0; i< Size; i++) ∥如果字符小于字符0,或者大于字符9,则返回0 if(ai]<ol ai>9) retum ∥走到这一步说明字符串由字符09组成,返回1 return l; 主进程源文件:man,c main.c #include"process, h Include <stdio. h> include <stdlib h> 甲 include< unistd. I> 甲 nclude< signal> 却 nclude <sys/ ypes,h include <sys/stat. h> 甲 include< tenth> int main(void) creat_and_run_moniter("proclistini") while(l) sleep(D) turn 以上内容是程序全部源文件
好的,我了解了您的问题。停车场管理系统是一个广泛应用于现代城市的管理软件,其主要功能是对停车场进行管理和监控。在C++程序设计中,可以使用面向对象的思想来设计该系统。 首先,我们需要定义一些基本的类,如车辆类、停车位类、停车场类和管理员类。然后,我们可以使用继承的方式,将这些类进行组合,以实现更复杂的功能。 在车辆类中,我们可以定义车辆的基本属性,如车牌号、车型等信息,并实现一些基本的方法,如进入停车场、离开停车场等。在停车位类中,我们可以定义停车位的状态信息,如是否空闲等,并实现一些基本的方法,如占用停车位、释放停车位等。 在停车场类中,我们可以使用容器类来管理车辆和停车位,实现车辆的进出管理和停车位的分配管理。在管理员类中,我们可以定义管理员的基本属性,并实现一些基本的方法,如查看停车场状态、添加/删除车辆等。 在程序实现中,我们可以使用文件输入输出流来保存停车场状态信息,并使用图形界面来实现用户交互。同时,我们还需要考虑一些异常情况的处理,如停车位已满、车牌号重复等情况的处理。 总之,停车场管理系统的C++程序设计需要考虑到系统的整体架构和功能实现,以及异常情况的处理。通过细致的设计和实现,我们可以打造出一个高效、稳定的停车场管理系统。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值