《完美C/C++》之windows守护进程

#include <stdio.h>
#include <stdlib.h>
#include <Windows.h>
#include <io.h>
#include <iostream>
LPCTSTR szAppClassName  = TEXT("Print Service");
LPCTSTR szAppWindowName = TEXT("Print Service");
using namespace std;

//隐藏DOS黑窗口
#pragma comment(linker,"/subsystem:\"windows\"  /entry:\"mainCRTStartup\"" )

//定义路径最大程度
#define MAX_PATH_NUM 4096
//定义守护进程名称
#define PROCCESS_NAME "printServer.exe"
//定义写入的注册表路径
#define SELFSTART_REGEDIT_PATH "Software\\Microsoft\\Windows\\CurrentVersion\\Run\\"

//设置本身开机自启动
BOOL SetSelfStart()
{
    //获取程序完整名称
    char pName[MAX_PATH_NUM]={0};
    GetModuleFileNameA(NULL,pName,MAX_PATH_NUM);

    //在注册表中写入启动信息
    HKEY hKey=NULL;
    LONG lRet=0;
    lRet=RegOpenKeyExA(HKEY_CURRENT_USER,SELFSTART_REGEDIT_PATH,0,KEY_ALL_ACCESS,&hKey);

    //判断是否成功
    if(lRet!=ERROR_SUCCESS)
    {
        return FALSE;
    }

    lRet=RegSetValueExA(hKey,"testProtect",0,REG_SZ,(const unsigned char*)pName,strlen(pName)+sizeof(char));

    //判断是否成功
    if(lRet!=ERROR_SUCCESS)
    {
        return FALSE;
    }

    //关闭注册表
    RegCloseKey(hKey);
    return TRUE;
}

int main()
{
    //获得守护的进程的句柄
    HWND handle = FindWindow(NULL, szAppWindowName);
    if (handle != NULL)
    {
        MessageBox(NULL, TEXT("Application is already running"), szAppClassName, MB_ICONERROR);
        ExitProcess(1);
    }

    //设置程序开机自启动
    if(!SetSelfStart())
    {
        cout<<"守护进程开机自启动失败"<<endl;
        return -1;
    }

    STARTUPINFOA si;
    //进程对象
    PROCESS_INFORMATION pi;
    //初始化
    ZeroMemory(&si,sizeof(si));
    si.cb=sizeof(si);
    ZeroMemory(&pi,sizeof(pi));

    //获取当前程序的路径
    char pPath[MAX_PATH_NUM]={0};
    GetCurrentDirectoryA(MAX_PATH_NUM,pPath);

    //拼接需要守护的程序
    strcat(pPath,"\\");
    strcat(pPath,PROCCESS_NAME);

    //构造cmd执行守护进程的字符串
    char pCmd[MAX_PATH_NUM]={0};
    //strcat(pCmd,"cmd /c ");
    strcat(pCmd,pPath);

    //无限循环,监视守护进程
    do{
        //检查守护程序是否存在
        if(_access(pPath,0)!=-1)
        {
            //创建子进程,判断是否执行成功
            if(!CreateProcessA(NULL,pCmd,NULL,NULL,FALSE,0,NULL,NULL,&si,&pi))
            {
                cout<<"守护进程启动失败,程序即将退出"<<endl;
                return -1;
            }

            //启动成功,获取进程的ID
            cout<<"守护进程成功,ID:"<<pi.dwProcessId<<endl;
            //无限等待子进程退出
            WaitForSingleObject(pi.hProcess,INFINITE);
            //如果退出了
            cout<<"守护进程退出了。。。"<<endl;
            //关闭进程和句柄
            CloseHandle(pi.hProcess);
            CloseHandle(pi.hThread);
        }
        else
        {
            cout<<"守护程序不存在"<<endl;
        }
        //睡一下,重启
        Sleep(1000);
    }
    while(true);
    return 0;
}

但这个里面

  HWND handle = FindWindow(NULL, szAppWindowName);

这部分没有太懂用法,我这边是做了一个放置重复启动的做法

改为了

if (IsAlreadyRun())
    {
        printf("程序已运行!");
        return 0;
    }


BOOL IsAlreadyRun()
{
    HANDLE hMutex = NULL;
    hMutex = CreateMutex(NULL, FALSE, “testProtect”);
    if (hMutex != NULL)
    {
        if (ERROR_ALREADY_EXISTS == GetLastError())
        {
            ReleaseMutex(hMutex);
            return TRUE;
        }
    }
    return FALSE;
}

### 回答1: 为了将一个进程转化为守护进程,我们需要进行以下几个步骤: 1. 创建一个子进程,并使用fork()系统调用来实现。子进程将从父进程继承所有的打开文件描述符以及其他资源。 2. 在子进程中调用setsid()系统调用,在新的会话中创建一个新的进程组。这将使得子进程脱离原来的终端会话。 3. 再次使用fork()系统调用创建第二个子进程,并在此子进程中调用exit()函数退出第一个子进程。这样做是为了避免第一个子进程重新获取控制终端会话。 4. 改变当前的工作目录到一个不会被卸载的目录,比如根目录 "/". 5. 关闭所有未使用的文件描述符。这可以通过遍历整数范围内的所有可能的文件描述符,并调用close()函数来实现。 6. 重新打开标准输入、标准输出和标准错误输出,将它们与 /dev/null 文件关联起来。这样做是为了防止在守护进程运行时产生不必要的输出。 7. 在守护进程中执行主要的任务,比如监听特定端口、处理请求等。 8. 必要时写入日志文件,以记录守护进程的活动和错误信息。 通过以上步骤,我们可以将一个进程转为守护进程守护进程在后台运行,独立于终端会话,并且能够更好地处理服务和应用程序。 ### 回答2: 编写一个守护进程的主要目的是使进程在后台运行,不受终端会话的影响,并且能够自动重启。在Linux C/C++开发中,可以通过以下步骤创建一个守护进程: 1. 创建一个子进程,并通过调用fork()函数复制父进程。 2. 在子进程中调用setsid()函数,创建一个新的会话,并使该进程成为该会话的领头进程,这样就能摆脱终端的控制。 3. 为了避免进程意外打开终端,首先需要在子进程中调用fork()函数,然后让父进程退出。这样子进程不会成为一个会话的领头进程。 4. 通过调用chdir()函数,将当前工作目录更改为根目录,这样可以避免占用其他文件系统导致其无法卸载。 5. 关闭不再使用的文件描述符,包括标准输入、标准输出和标准错误输出,可以通过调用close()函数关闭。 6. 为了确保程序能够自动重启,在守护进程的主循环中使用无限循环,可以通过调用sleep()函数让进程间歇性地休眠。 总结:上述步骤可以用来创建一个基本的守护进程,使进程在后台运行,并能够自动重启。当然,根据实际需求,还可以对进程进行更多的配置和调整,以适应特定的业务需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值