windows服务开发
admin
2024-03-24 02:30:32
0

参考博客:
C++调用StartService启动服务失败1053分析与解决
StartServiceCtrlDispatcher函数
指定的服务已标记为删除
安装和卸载服务

windows服务开发分为两个步骤:

  1. 编写服务程序.exe
  2. 安装服务程序,使在services.msc里面可以看见自己的服务

编写服务程序

问:为什么需要编写服务程序?是不是随便一个exe都可以作为服务程序运行?
答:服务程序必须主动调用StartServiceCtrlDispatcher上报自己的当前状态,否则服务控制台由于不知道服务的运行状态的而报错,例如:StartService启动服务失败1053。所以开发服务程序的时候我们必须通过StartServiceCtrlDispatcher上报状态并且处理服务事件。

  1. 调用StartServiceCtrlDispatcher,设置服务事件回调。如果程序作为非服务状态运行则会返回错误:ERROR_FAILED_SERVICE_CONTROLLER_CONNECT,程序会一直阻塞运行,直至SERVICE_STOPPED状态才返回
  2. 注册并相应服务控制事件,例如服务控制台停止事件
  3. 上报服务当前状态

CMscSvc.h

//如果作为服务程序,需要向服务报告自己的状态,否则服务控制台是不知道服务状态,会失败
class CMscSvc
{
public://如果作为服务程序,需要注册服务static void RegisterSvc(std::string msc_name);//上报启动成功事件static void ReportStart();//上报停止成功事件static void ReportStop();
private://如果作为服务程序需要报告自己的状态static void ReportSvcStatus(DWORD dwCurrentState, DWORD dwWin32ExitCode, DWORD dwWaitHint);//服务主处理函数,内部添加自己的业务逻辑代码。可包装到doWork函数内部static VOID WINAPI SvcMain(DWORD dwArgc, LPTSTR *lpszArgv);//响应服务控制台事件,例如停止static VOID WINAPI SvcCtrlHandler(DWORD dwCtrl);static SERVICE_STATUS_HANDLE s_gSvcStatusHandle;static SERVICE_STATUS s_gSvcStatus;static std::string m_msc_name;
};

CMscSvc.cpp


SERVICE_STATUS_HANDLE CMscSvc::s_gSvcStatusHandle;
SERVICE_STATUS CMscSvc::s_gSvcStatus;
std::string CMscSvc::m_msc_name = "";void CMscSvc::RegisterSvc(std::string msc_name)
{m_msc_name = msc_name;char szName[MAX_PATH] = "";_stprintf_s(szName, "%s", m_msc_name.c_str());SERVICE_TABLE_ENTRY DispatchTable[] ={{ szName, (LPSERVICE_MAIN_FUNCTION)SvcMain },{ NULL, NULL }};if (!StartServiceCtrlDispatcher(DispatchTable)){DWORD dwErrcode = GetLastError();DebugPrint(_T("StartServiceCtrlDispatcher failed,errcode:%d"), dwErrcode);}
}void CMscSvc::ReportStart()
{ReportSvcStatus(SERVICE_RUNNING, 0, 0);
}void CMscSvc::ReportStop()
{ReportSvcStatus(SERVICE_STOPPED, 0, 0);
}VOID WINAPI CMscSvc::SvcMain(DWORD dwArgc, LPTSTR *lpszArgv)
{// Register the handler function for the service s_gSvcStatusHandle = RegisterServiceCtrlHandler(m_msc_name.c_str(), SvcCtrlHandler);if (!s_gSvcStatusHandle){DWORD dwErrcode = GetLastError();DebugPrint(_T("RegisterServiceCtrlHandler failed,errcode:%d"), dwErrcode);return;}// These SERVICE_STATUS members remain as set here s_gSvcStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;s_gSvcStatus.dwServiceSpecificExitCode = 0;// Report initial status to the SCMReportSvcStatus(SERVICE_START_PENDING, NO_ERROR, 3000);//上报启动成功状态ReportStart();//开始处理业务doWork();
}void CMscSvc::ReportSvcStatus(DWORD dwCurrentState, DWORD dwWin32ExitCode, DWORD dwWaitHint)
{static DWORD dwCheckPoint = 1;// Fill in the SERVICE_STATUS structure.s_gSvcStatus.dwCurrentState = dwCurrentState;s_gSvcStatus.dwWin32ExitCode = dwWin32ExitCode;s_gSvcStatus.dwWaitHint = dwWaitHint;if (dwCurrentState == SERVICE_START_PENDING)s_gSvcStatus.dwControlsAccepted = 0;else s_gSvcStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP;if ((dwCurrentState == SERVICE_RUNNING) ||(dwCurrentState == SERVICE_STOPPED))s_gSvcStatus.dwCheckPoint = 0;else s_gSvcStatus.dwCheckPoint = dwCheckPoint++;SetServiceStatus(s_gSvcStatusHandle, &s_gSvcStatus);
}//每当使用controlService函数向服务发送控制代码时,由SCM调用
VOID WINAPI CMscSvc::SvcCtrlHandler(DWORD dwCtrl)
{// Handle the requested control code. switch (dwCtrl){case SERVICE_CONTROL_STOP:{ReportSvcStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);//退出业务处理逻辑exitWork()//上报停止状态成功ReportStop()return;}case SERVICE_CONTROL_INTERROGATE:break;default:break;}
}

调用

void _main()
{CMscSvc::RegisterSvc("testSvr");
}

编写服务安装和卸载程序

注意步骤:

  1. 打开服务管理器OpenSCManager
  2. 创建或者打开服务:CreateService OpenService
  3. 启动服务:StartService
  4. 查询服务状态:QueryServiceStatus

CMsc.h

class CMsc
{
public:CMsc(std::string msc_name,std::string exe_full_path,std::string exe_param = "",std::string msc_desc = "");~CMsc();BOOL start();BOOL stop(BOOL bUnInstall = FALSE/*是否卸载*/);
private:std::string m_msc_name;std::string m_msc_display_name;std::string m_msc_desc;std::string m_exe_cmd;
};

CMsc.cpp

class CSafeSCHandle
{
public:CSafeSCHandle(const SC_HANDLE& hscm) :m_hscm(hscm){};~CSafeSCHandle(){if (m_hscm){CloseServiceHandle(m_hscm);}}operator const SC_HANDLE&(){ return m_hscm; }
private:const SC_HANDLE &m_hscm;
};CMsc::CMsc(std::string msc_name,std::string exe_full_path, std::string exe_param)
{m_msc_name = msc_name;m_msc_display_name = m_msc_name;m_exe_cmd = exe_full_path + _T(" ") + exe_param;m_msc_desc = msc_desc;DebugPrint("CMsc name:%s,exe_full_path:%s,exe_param:%s", m_msc_name.c_str(), exe_full_path.c_str(), exe_param.c_str());
}CMsc::~CMsc()
{
}BOOL CMsc::start()
{//打开msc服务管理器CSafeSCHandle hScm(OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS));if (hScm == NULL){DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,start,OpenSCManager failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}//如果服务已经存在则直接startCSafeSCHandle hService(OpenService(hScm, m_msc_name.c_str(), SERVICE_ALL_ACCESS));if (hService != NULL){SERVICE_STATUS status;QueryServiceStatus(hService, &status);if (status.dwCurrentState == SERVICE_RUNNING){return TRUE;}if (StartService(hService, 0, NULL)){DebugPrint("CMsc name:%s,start,StartService OK", m_msc_name.c_str());return TRUE;}else{DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,start,StartService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}}//如果服务不存在则创建CSafeSCHandle hCService(CreateService(hScm, m_msc_name.c_str(), m_msc_display_name.c_str(), SERVICE_ALL_ACCESS,SERVICE_WIN32_OWN_PROCESS, SERVICE_AUTO_START, SERVICE_ERROR_NORMAL, m_exe_cmd.c_str(),NULL, NULL, "", NULL, ""));if (hCService == NULL){DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,start,CreateService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}//设置服务描述if (!m_msc_desc.empty()){SERVICE_DESCRIPTION sd;sd.lpDescription = (char *)m_msc_desc.c_str();ChangeServiceConfig2(hCService, SERVICE_CONFIG_DESCRIPTION, &sd);}SERVICE_STATUS status;QueryServiceStatus(hCService, &status);if (status.dwCurrentState == SERVICE_RUNNING){DebugPrint("CMsc name:%s,start,StartService OK,is already runing", m_msc_name.c_str());return TRUE;}if (StartService(hCService, 0, NULL)){DebugPrint("CMsc name:%s,start,StartService OK", m_msc_name.c_str());return TRUE;}else{DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,start,StartService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}
}BOOL CMsc::stop(BOOL bUnInstall)
{//打开msc服务管理器CSafeSCHandle hScm(OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS));if (hScm == NULL){DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,stop,OpenSCManager failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}//如果服务已经存在则直接startCSafeSCHandle hService(OpenService(hScm, m_msc_name.c_str(), SERVICE_ALL_ACCESS));if (hService == NULL){DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,stop,OpenService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}//查询服务状态SERVICE_STATUS status;QueryServiceStatus(hService, &status);if (status.dwCurrentState == SERVICE_RUNNING){//停止服务ControlService(hService, SERVICE_CONTROL_STOP, &status);if (status.dwCurrentState != NO_ERROR){DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,stop,ControlService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);return FALSE;}}//如果卸载服务if (bUnInstall && status.dwCurrentState == SERVICE_STOPPED){if (DeleteService(hService)){DebugPrint("CMsc name:%s,stop,DeleteService OK", m_msc_name.c_str());}else{DWORD dwErrcode = GetLastError();DebugPrint("CMsc name:%s,stop,DeleteService failed,errcode:%d", m_msc_name.c_str(), dwErrcode);}}DebugPrint("CMsc name:%s,stop,ControlService OK", m_msc_name.c_str());return TRUE;
}

调用

void _main()
{//msc_name-服务名称//exe_full_path exe的全路径//exe_params exe的执行参数CMsc msc(msc_name, exe_full_path, "exe_params");//安装并启动服务msc.start();//停止并且卸载服务msc.stop(TRUE);}

此时我们就可以安装服务程序,并且在services.msc中查看了。

Note:如果服务管理器打开了,则删除服务管理器会失败[指定的服务已标记为删除]。必须关闭重试。

相关内容

热门资讯

【MySQL】锁 锁 文章目录锁全局锁表级锁表锁元数据锁(MDL)意向锁AUTO-INC锁...
【内网安全】 隧道搭建穿透上线... 文章目录内网穿透-Ngrok-入门-上线1、服务端配置:2、客户端连接服务端ÿ...
GCN的几种模型复现笔记 引言 本篇笔记紧接上文,主要是上一篇看写了快2w字,再去接入代码感觉有点...
数据分页展示逻辑 import java.util.Arrays;import java.util.List;impo...
Redis为什么选择单线程?R... 目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程?三、R...
【已解决】ERROR: Cou... 正确指令: pip install pyyaml
关于测试,我发现了哪些新大陆 关于测试 平常也只是听说过一些关于测试的术语,但并没有使用过测试工具。偶然看到编程老师...
Lock 接口解读 前置知识点Synchronized synchronized 是 Java 中的关键字,...
Win7 专业版安装中文包、汉... 参考资料:http://www.metsky.com/archives/350.htm...
3 ROS1通讯编程提高(1) 3 ROS1通讯编程提高3.1 使用VS Code编译ROS13.1.1 VS Code的安装和配置...
大模型未来趋势 大模型是人工智能领域的重要发展趋势之一,未来有着广阔的应用前景和发展空间。以下是大模型未来的趋势和展...
python实战应用讲解-【n... 目录 如何在Python中计算残余的平方和 方法1:使用其Base公式 方法2:使用statsmod...
学习u-boot 需要了解的m... 一、常用函数 1. origin 函数 origin 函数的返回值就是变量来源。使用格式如下...
常用python爬虫库介绍与简... 通用 urllib -网络库(stdlib)。 requests -网络库。 grab – 网络库&...
药品批准文号查询|药融云-中国... 药品批文是国家食品药品监督管理局(NMPA)对药品的审评和批准的证明文件...
【2023-03-22】SRS... 【2023-03-22】SRS推流搭配FFmpeg实现目标检测 说明: 外侧测试使用SRS播放器测...
有限元三角形单元的等效节点力 文章目录前言一、重新复习一下有限元三角形单元的理论1、三角形单元的形函数(Nÿ...
初级算法-哈希表 主要记录算法和数据结构学习笔记,新的一年更上一层楼! 初级算法-哈希表...
进程间通信【Linux】 1. 进程间通信 1.1 什么是进程间通信 在 Linux 系统中,进程间通信...
【Docker】P3 Dock... Docker数据卷、宿主机与挂载数据卷的概念及作用挂载宿主机配置数据卷挂载操作示例一个容器挂载多个目...