这次我们来创建一个windows本地服务,需要有以下功能:

  • 安装服务。
  • 卸载服务。
  • 手动启动服务。
  • 开机自动启动服务。
  • 控制服务(停止、暂停、恢复、启动)。

服务概念及介绍

看下图,一切尽在不言中了(-_-):

安装服务并开机启动

  • 代码逻辑: 打开SCM(Service Control Manager)-> 创建服务。

  • 在CreateService调用中,我们传入SERVICE_AUTO_START里表明这个服务是开启自启动的,SCM会在开机时调用StartService来启动我们的服务;同时我们传入NULL作为服务开始名称,这样该服务就可以开机启动为系统服务。在调用installService后,我们会调用startService来手动启动服务。

  • 这里我用SAFE_CALL简化了错误处理逻辑。

    wstring getExeFullFilename()
    {
    static wchar_t buffer[1024]; SAFE_CALL(GetModuleFileNameW(NULL, buffer, 1024), 0);
    return wstring(buffer);
    } void installService()
    {
    auto scmHandle = OpenSCManagerW(NULL, NULL, SC_MANAGER_CREATE_SERVICE);
    SAFE_CALL(scmHandle, NULL); auto serviceHandle = CreateServiceW(scmHandle,
    L"lgxZJ::Service",
    L"lgxZJ::Service",
    SERVICE_ALL_ACCESS,
    SERVICE_WIN32_OWN_PROCESS,
    SERVICE_AUTO_START,
    SERVICE_ERROR_NORMAL,
    getExeFullFilename().c_str(),
    NULL, NULL, L"", NULL, L"");
    SAFE_CALL(serviceHandle, NULL); CloseServiceHandle(scmHandle);
    CloseServiceHandle(serviceHandle);
    } void startService()
    {
    auto scmHandle = OpenSCManagerW(NULL, NULL, SC_MANAGER_ALL_ACCESS);
    SAFE_CALL(scmHandle, NULL); auto serviceHandle = OpenServiceW( scmHandle,
    L"lgxZJ::Service",
    SERVICE_ALL_ACCESS);
    SAFE_CALL(serviceHandle, NULL); SERVICE_STATUS serviceStatus;
    SAFE_CALL(QueryServiceStatus(serviceHandle, &serviceStatus), 0);
    if (serviceStatus.dwCurrentState == SERVICE_START &&
    serviceStatus.dwCurrentState != SERVICE_START_PENDING)
    return; SAFE_CALL(StartServiceW(serviceHandle, 0, NULL), FALSE); CloseServiceHandle(scmHandle);
    CloseServiceHandle(serviceHandle);
    } #define SAFE_CALL(FuncCall, ErrorCode) \
    if (FuncCall == ErrorCode) { \
    cout << #FuncCall " error, code:" << GetLastError() \
    << " ,line:" << \__LINE__ << "\n"; \
    exit(-1); \
    }

卸载服务

  • 代码逻辑: 打开SCM(Service Control Manager)-> 打开服务 -> 停止服务(如果正在运行) -> 删除服务。

    void uninstallService()
    {
    auto scmHandle = OpenSCManagerW(NULL, NULL, SC_MANAGER_ALL_ACCESS);
    SAFE_CALL(scmHandle, NULL); auto serviceHandle = OpenServiceW( scmHandle,
    L"lgxZJ::Service",
    SERVICE_ALL_ACCESS);
    SAFE_CALL(serviceHandle, NULL); SERVICE_STATUS serviceStatus;
    SAFE_CALL(QueryServiceStatus(serviceHandle, &serviceStatus), 0);
    if (serviceStatus.dwCurrentState == SERVICE_RUNNING) {
    SAFE_CALL(ControlService(serviceHandle, SERVICE_CONTROL_STOP, &serviceStatus), 0);
    SAFE_CALL(serviceStatus.dwCurrentState, NO_ERROR); do {
    SAFE_CALL(QueryServiceStatus(serviceHandle, &serviceStatus), 0);
    Sleep(1000);
    } while (serviceStatus.dwCurrentState != SERVICE_STOPPED);
    } SAFE_CALL(DeleteService(serviceHandle), FALSE); CloseServiceHandle(scmHandle);
    CloseServiceHandle(serviceHandle);
    }

手动启动服务

  • 代码逻辑: 打开SCM -> 打开服务 -> 启动服务。

    void startService()
    {
    auto scmHandle = OpenSCManagerW(NULL, NULL, SC_MANAGER_ALL_ACCESS);
    SAFE_CALL(scmHandle, NULL); auto serviceHandle = OpenServiceW( scmHandle,
    L"lgxZJ::Service",
    SERVICE_ALL_ACCESS);
    SAFE_CALL(serviceHandle, NULL); SERVICE_STATUS serviceStatus;
    SAFE_CALL(QueryServiceStatus(serviceHandle, &serviceStatus), 0);
    if (serviceStatus.dwCurrentState == SERVICE_START &&
    serviceStatus.dwCurrentState != SERVICE_START_PENDING)
    return; SAFE_CALL(StartServiceW(serviceHandle, 0, NULL), FALSE); CloseServiceHandle(scmHandle);
    CloseServiceHandle(serviceHandle);
    }

运行服务(服务启动时会运行服务)、服务控制处理

  • 代码逻辑: 启动分发器(连接到SCM) -> 注册服务控制处理器 -> 在控制处理器中对服务控制进行处理(通过SetServiceStatus反馈服务状态和设置接受的控制)。

    void runService()
    {
    const SERVICE_TABLE_ENTRYW serviceTable[] = {
    { L"", ServiceMain },
    { NULL, NULL }
    }; SAFE_CALL(StartServiceCtrlDispatcherW(&serviceTable[0]), 0);
    } SERVICE_STATUS_HANDLE g_serviceStatusHandle = NULL; void setServiceStatus(DWORD status)
    {
    SERVICE_STATUS serviceStatus;
    serviceStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
    serviceStatus.dwWin32ExitCode = NO_ERROR;
    serviceStatus.dwServiceSpecificExitCode = 0;
    serviceStatus.dwWaitHint = 2000;
    serviceStatus.dwCheckPoint = 0;
    serviceStatus.dwControlsAccepted = SERVICE_ACCEPT_PAUSE_CONTINUE |
    SERVICE_ACCEPT_SHUTDOWN |
    SERVICE_ACCEPT_STOP; serviceStatus.dwCurrentState = status;
    SAFE_CALL(SetServiceStatus(g_serviceStatusHandle, &serviceStatus), 0);
    } VOID WINAPI ServiceHandler(DWORD controlCode)
    {
    switch (controlCode)
    {
    case SERVICE_CONTROL_CONTINUE:
    setServiceStatus(SERVICE_START_PENDING); break;
    case SERVICE_CONTROL_INTERROGATE:
    break;
    case SERVICE_CONTROL_PAUSE:
    setServiceStatus(SERVICE_PAUSED); break;
    case SERVICE_CONTROL_SHUTDOWN:
    setServiceStatus(SERVICE_STOPPED); break;
    case SERVICE_CONTROL_STOP:
    setServiceStatus(SERVICE_STOPPED); break;
    default:
    break;
    }
    } VOID WINAPI ServiceMain(DWORD argc, LPWSTR *argv)
    {
    g_serviceStatusHandle = RegisterServiceCtrlHandlerW(L"lgxZJ::Service", &ServiceHandler);
    if (g_serviceStatusHandle == 0)
    {
    cout << "RegisterServiceCtrlHandlerW error, code:" << GetLastError()
    << " ,line:" << __LINE__ << "\n";
    exit(-1);
    } setServiceStatus(SERVICE_START_PENDING);
    setServiceStatus(SERVICE_RUNNING);
    }

完整代码

生成的exe需要以管理员权限启动,完整代码见此处

创建Windows服务(C++)的更多相关文章

  1. 用C#创建Windows服务(Windows Services)

    用C#创建Windows服务(Windows Services) 学习:  第一步:创建服务框架 创建一个新的 Windows 服务项目,可以从Visual C# 工程中选取 Windows 服务(W ...

  2. 玩转Windows服务系列——创建Windows服务

    创建Windows服务的项目 新建项目->C++语言->ATL->ATL项目->服务(EXE) 这样就创建了一个Windows服务项目. 生成的解决方案包含两个项目:Servi ...

  3. .Net创建windows服务入门

    本文主要记录学习.net 如何创建windows服务. 1.创建一个Windows服务程序 2.新建安装程序 3.修改service文件 代码如下 protected override void On ...

  4. C# 创建Windows服务

    创建windows服务项目   2 右键点击Service1.cs,查看代码, 用于编写操作逻辑代码 3 代码中OnStart用于执行服务事件,一般采用线程方式执行方法,便于隔一段事件执行一回 END ...

  5. 使用Topshelf创建Windows服务

    概述 Topshelf是创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps with Topshelf通过5个步骤详细的 ...

  6. [转]C#创建Windows服务与安装

    本文档用于创建windows服务说明,使用vs2010系统平台 创建项目 1 创建windows服务项目 2 右键点击Service1.cs,查看代码, 用于编写操作逻辑代码 3 代码中OnStart ...

  7. [Solution] Microsoft Windows 服务(2) 使用Topshelf创建Windows服务

    除了通过.net提供的windows服务模板外,Topshelf是创建Windows服务的另一种方法. 官网教程:http://docs.topshelf-project.com/en/latest/ ...

  8. 在64位windows下使用instsrv.exe和srvany.exe创建windows服务

    在64位windows下使用instsrv.exe和srvany.exe创建windows服务   在32位的windows下,包括windows7,windows xp以及windows 2003, ...

  9. 使用Topshelf 5步创建Windows 服务 z

    使用Topshelf创建Windows 服务简要的介绍了创建Windows服务的另一种方法,老外的一篇文章Create a .NET Windows Service in 5 steps with T ...

  10. C#创建windows服务搭配定时器Timer使用实例(用代码做,截图版)

       功能说明:C#创建一个windows服务,服务启动时D:\mcWindowsService.txt写入数据,服务运行期间每隔两秒写入当前时间. 原理这些就不说了,三语两语说不清楚,直接贴一个实例 ...

随机推荐

  1. Android系统--输入系统(十五)实战_使用GlobalKey一键启动程序

    Android系统--输入系统(十五)实战_使用GlobalKey一键启动程序 1. 一键启动的过程 1.1 对于global key, 系统会根据global_keys.xml发送消息给某个组件 & ...

  2. Aspose.words 书签定位

    1. 简介 Aspose.words 可以在不使用 Microsoft.Word 的情况下生成.修改.转换.打印文档.不依赖office组件,这一点给我们提供了极大的便利性,可以简单的引入 DLL(D ...

  3. Swift语法初见

    Swift语法初见 http://c.biancheng.net/cpp/html/2424.html 类型的声明: let implicitInteger = 70 let implicitDoub ...

  4. Java之StringBuffer,StringBuilder,Math,Date,SimpleDateFormat,UUID,File

    java.lang 类 StringBuffer java.lang.Object java.lang.StringBuffer 所有已实现的接口: Serializable, Appendable, ...

  5. If I were you

    别让孩子输在起跑线上,这句话有道理吗? 答:和别人比较是人生悲剧的源头. 做哪些事情可以提高生活的品质? 答:定期扔东西. 结婚以后,两个人在一起最重要的是什么? 答:就当这婚还没结呢. 去做你害怕的 ...

  6. 解决win10注册错误 错误代码0x8002801c

    使用win10的过程中经常碰到各种注册错误,让人抓狂!!! 现在分享一个完美的解决方法(非原创): 最简洁的办法是:1.自行将msinet.ocx(win10系统64位)组件复制到C:\Windows ...

  7. 安装Scala-2.11.7——集群学习日记

    前言 在安装Spark之前,我们需要安装Scala语言的支持.在此我选择的是scala-2.11.7版本. scala-2.11.7下载 为了方便,我现在我的SparkMaster主机上先安装,把目录 ...

  8. [JAVASCRIPT]实现页面复制至电脑剪贴板

    一. 方法 方1: window.clipboarddata  可惜不支持chrome , chrome 下会提示找不到 clipboarddata 对象 方2: 采用国外大牛写的ZeroClipbo ...

  9. 【python】字符排序

    一.摘要 最近在做一个排序的东西,被python的字符串编码格式折腾了一会儿,总结下 二.排序 英文排序不用说,sort sorted 比较好,内部已经实现 主要是中文,方法是查表获取拼音再进行排序. ...

  10. 【Django】学习资料

    一.基础 http://www.ibm.com/developerworks/cn/linux/l-django/ 安装.数据库连接.url路由转发 http://djangobook.py3k.cn ...