原文:玩转Windows服务系列——Windows服务小技巧

伴随着研究Windows服务,逐渐掌握了一些小技巧,现在与大家分享一下。

将Windows服务转变为控制台程序

由于默认的Windows服务程序,编译后为Win32的窗口程序。我们在程序启动或运行过程中,如果想看到一些调试信息,那么就只能通过DebugView或者输出到日志的方式了。因为如果我们通过printf或者std::cout输出调试信息的话,Win32窗口程序是无法显示的。

此时,我们是多么怀念我们的经典的控制台程序啊,它可以很方便的将我们的调试信息输出出来,简直是太方便了。既然如此,那我们就让它一秒钟变格格吧,额,应该是一秒钟变控制台。

下面分享一下我的实现代码

#ifdef _DEBUG
//Debug版本,直接输出到控制台
#define OUT(s) printf_s(s);
#define OUT_LN(s) printf_s(s##"\r\n");
#else
//非Debug版本,则输出到调试器,一般使用DebugView
#define OUT(s) OutputDebugString(s);
#define OUT_LN(s) OutputDebugString(s);
#endif class CServicesModule : public ATL::CAtlServiceModuleT< CServicesModule, IDS_SERVICENAME >
{
public :
DECLARE_LIBID(LIBID_ServicesLib)
DECLARE_REGISTRY_APPID_RESOURCEID(IDR_SERVICES, "{0794CF96-5CC5-432E-8C1D-52B980ACBE0F}")
HRESULT InitializeSecurity() throw()
{
return S_OK;
}
//服务启动
HRESULT Load();
//服务停止
HRESULT UnLoad(); HRESULT Run(_In_ int nShowCmd = SW_HIDE) throw()
{
HRESULT hr = S_OK;
OUT_LN("准备启动服务");
hr = Load();
if(hr)
{
OUT_LN("启动服务失败");
return hr;
}
OUT_LN("Services服务已启动"); hr = ATL::CAtlServiceModuleT< CServicesModule, IDS_SERVICENAME >::Run(nShowCmd); hr = UnLoad();
OUT_LN("Services服务正常退出");
return hr;
}
}; CServicesModule _AtlModule; //
#ifndef _DEBUG
//非Debug版本,编译为Win32程序
extern "C" int WINAPI _tWinMain(HINSTANCE /*hInstance*/, HINSTANCE /*hPrevInstance*/,
LPTSTR /*lpCmdLine*/, int nShowCmd)
{
return _AtlModule.WinMain(nShowCmd);
}
#else
//Debug版本,编译为控制台程序
int _tmain(int argc, _TCHAR* argv[])
{
return _AtlModule.WinMain(SW_SHOW);
}
#endif HRESULT CServicesModule::Load()
{
OUT_LN("服务正在启动");
return 0;
} HRESULT CServicesModule::UnLoad()
{
OUT_LN("服务正在停止");
return 0;
}

通过_DEBUG宏来区分是否编译成控制台程序。

当指定编译Debug版本时,可以将程序编译为控制台程序,通过RegServer注册服务,然后直接运行服务exe程序,这样通过printf输出的信息,就可以在控制台上显示了,如下图。

当指定编译Release版本时,将程序编译为Win32程序,通过Service注册服务,通过服务管理器管理服务的运行和停止。

当然,这还不是全部,还有一个地方需要设置,下面分别是Debug和Release下的设置

 

当然,还有一种更简单的方法,可以将Debug和Release模式下的“子系统”项修改为“未设置”。这样编译器在编译链接时,会根据代码中的入口函数,自动将代码链接为对应的程序。如图

 

注册服务为自动启动服务

注册服务的流程已经在前面的博客玩转Windows服务系列——Debug、Release版本的注册和卸载,及其原理中有介绍,这里就不再多说。重点说一下创建服务的Windows API CreateService。

如下是MSDN中的API声明

SC_HANDLE WINAPI CreateService(
_In_ SC_HANDLE hSCManager,
_In_ LPCTSTR lpServiceName,
_In_opt_ LPCTSTR lpDisplayName,
_In_ DWORD dwDesiredAccess,
_In_ DWORD dwServiceType,
_In_ DWORD dwStartType,
_In_ DWORD dwErrorControl,
_In_opt_ LPCTSTR lpBinaryPathName,
_In_opt_ LPCTSTR lpLoadOrderGroup,
_Out_opt_ LPDWORD lpdwTagId,
_In_opt_ LPCTSTR lpDependencies,
_In_opt_ LPCTSTR lpServiceStartName,
_In_opt_ LPCTSTR lpPassword
);

参数太多,不一一介绍,详细介绍可以查看MSDN。

其中第六个参数,代表启动方式

dwStartType [in]
The service start options. This parameter can be one of the following values. Value Meaning
SERVICE_AUTO_START
0x00000002 A service started automatically by the service control manager during system startup. For more information, see Automatically Starting Services. SERVICE_BOOT_START
0x00000000 A device driver started by the system loader. This value is valid only for driver services. SERVICE_DEMAND_START
0x00000003 A service started by the service control manager when a process calls the StartService function. For more information, see Starting Services on Demand. SERVICE_DISABLED
0x00000004 A service that cannot be started. Attempts to start the service result in the error code ERROR_SERVICE_DISABLED. SERVICE_SYSTEM_START
0x00000001 A device driver started by the IoInitSystem function. This value is valid only for driver services.

SERVICE_AUTO_START 表示自动启动,这个参数就是我们想要的。

SERVICE_BOOT_START 也属于自动启动,但是只能用于内核服务。

SERVICE_DEMAND_START 手动启动,这是目前服务的默认启动方式。

SERVICE_DISABLED 禁止启动。

SERVICE_SYSTEM_START 属于自动启动,但只能用于内核服务。

所以,我们只需要在调用CreateService方法时,设置dwStartType参数为SERVICE_AUTO_START即可实现服务自动启动,而CreateService的其他参数,则参考现在的调用参数

::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
SERVICE_AUTO_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0"), NULL, NULL);

接下来,我们只需要重载命令行解析方法,添加参数用于确定是否自动启动即可。

 

注册服务时设置服务的依赖关系

设置服务的依赖关系仍然要看CreateService服务,这次我们看倒数第三个参数lpDependencies

lpDependencies [in, optional]
A pointer to a double null-terminated array of null-separated names of services or load ordering groups that the system must start before this service. Specify NULL or an empty string if the service has no dependencies. Dependency on a group means that this service can run if at least one member of the group is running after an attempt to start all members of the group. You must prefix group names with SC_GROUP_IDENTIFIER so that they can be distinguished from a service name, because services and service groups share the same name space.

lpDependencies是一个指针,指针指向了一个以’\0\0’结尾,并且以’\0’分割开的字符串,由’\0’分割开的字符串为依赖服务的名字。

比如,如果设置当前服务依赖RPCSS 和DependTest服务的,则可以这样调用CreateService方法

::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
SERVICE_AUTO_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0DependTest\0"), NULL, NULL);

由于字符串自身会以’\0’结尾,所以字符串内容中的结尾处只需写一个’\0’。

接下来,我们只需要重载命令行解析方法,添加参数用于确定是否自动启动即可

 

添加自定义命令行参数

添加自定义命令行参数 Auto, 用来设置启动方式为自动启动, 并且给Service参数添加依赖项,实现代码如下

// Services.cpp : WinMain 的实现

#include "stdafx.h"
#include "resource.h"
#include "Services_i.h" using namespace ATL; #include <stdio.h> #ifdef _DEBUG
#define OUT(s) printf_s(s);
#define OUT_LN(s) printf_s(s##"\r\n");
#else
#define OUT(s) OutputDebugString(s);
#define OUT_LN(s) OutputDebugString(s);
#endif class CServicesModule : public ATL::CAtlServiceModuleT< CServicesModule, IDS_SERVICENAME >
{
public :
DECLARE_LIBID(LIBID_ServicesLib)
DECLARE_REGISTRY_APPID_RESOURCEID(IDR_SERVICES, "{0794CF96-5CC5-432E-8C1D-52B980ACBE0F}")
HRESULT InitializeSecurity() throw()
{
// TODO : 调用 CoInitializeSecurity 并为服务提供适当的安全设置
// 建议 - PKT 级别的身份验证、
// RPC_C_IMP_LEVEL_IDENTIFY 的模拟级别
// 以及适当的非 NULL 安全描述符。 return S_OK;
}
//服务启动
HRESULT Load();
//服务停止
HRESULT UnLoad(); // Parses the command line and registers/unregisters the rgs file if necessary
bool ParseCommandLine(
_In_z_ LPCTSTR lpCmdLine,
_Out_ HRESULT* pnRetCode) throw();
//注册服务
BOOL Install() throw(); //重写此方法,主要是为了调用重写的Install方法
inline HRESULT RegisterAppId(_In_ bool bService = false) throw()
{
if (!Uninstall())
return E_FAIL; CServicesModule::UpdateRegistryAppId(TRUE); CRegKey keyAppID;
keyAppID.Open(HKEY_CLASSES_ROOT, _T("AppID"), KEY_WRITE); CRegKey key; key.Create(keyAppID, CServicesModule::GetAppIdT()); key.DeleteValue(_T("LocalService")); key.SetStringValue(_T("LocalService"), m_szServiceName); // Create service
if (!Install())
return E_FAIL;
return S_OK;
} HRESULT Run(_In_ int nShowCmd = SW_HIDE) throw()
{
HRESULT hr = S_OK;
OUT_LN("准备启动服务");
hr = Load();
if(hr)
{
OUT_LN("启动服务失败");
return hr;
}
OUT_LN("Services服务已启动"); hr = ATL::CAtlServiceModuleT< CServicesModule, IDS_SERVICENAME >::Run(nShowCmd); hr = UnLoad();
OUT_LN("Services服务正常退出");
return hr;
} private:
_TCHAR dependServices[256];
DWORD size;
DWORD dwStartType;
}; CServicesModule _AtlModule; //
#ifndef _DEBUG
extern "C" int WINAPI _tWinMain(HINSTANCE /*hInstance*/, HINSTANCE /*hPrevInstance*/,
LPTSTR /*lpCmdLine*/, int nShowCmd)
{
return _AtlModule.WinMain(nShowCmd);
}
#else
int _tmain(int argc, _TCHAR* argv[])
{
return _AtlModule.WinMain(SW_SHOW);
}
#endif HRESULT CServicesModule::Load()
{
memset(dependServices, 0, sizeof(dependServices));
dwStartType = SERVICE_DEMAND_START; OUT_LN("服务正在启动");
return 0;
} HRESULT CServicesModule::UnLoad()
{
OUT_LN("服务正在停止");
return 0;
} // Parses the command line and registers/unregisters the rgs file if necessary
bool CServicesModule::ParseCommandLine(
_In_z_ LPCTSTR lpCmdLine,
_Out_ HRESULT* pnRetCode) throw()
{
if (!CAtlExeModuleT<CServicesModule>::ParseCommandLine(lpCmdLine, pnRetCode))
return false; TCHAR szTokens[] = _T("-/");
*pnRetCode = S_OK;
LPCTSTR lpszToken = FindOneOf(lpCmdLine, szTokens);
while (lpszToken != NULL)
{
if (WordCmpI(lpszToken, _T("Service"))==0)
{
lpszToken += _tcslen(_T("Service"));
//循环读取依赖项
while (true)
{
LPCTSTR lpszTokenBegin = lpszToken;
while(isprint(*lpszToken) && *lpszToken != _T(' ') && *lpszToken != _T('-') && *lpszToken != _T('/'))
{
lpszToken++;
}
DWORD tokenSize = (lpszToken - lpszTokenBegin);
memcpy_s(dependServices + size, sizeof(dependServices) - size, lpszTokenBegin, tokenSize * sizeof(_TCHAR));
size += tokenSize;
dependServices[size] = _T('\0');
if(tokenSize)
{
size++;
} while(isprint(*lpszToken) && *lpszToken == ' ')
{
lpszToken++;
}
if((*lpszToken == _T('\0') || *lpszToken == _T('-') || *lpszToken == _T('/')))
{
dependServices[size + 1] = _T('\0');
break;
}
}
*pnRetCode = this->RegisterAppId(true);
if (SUCCEEDED(*pnRetCode))
*pnRetCode = this->RegisterServer(TRUE);
return false;
} //设置启动类型
if (WordCmpI(lpszToken, _T("Auto"))==0)
{
dwStartType = SERVICE_AUTO_START;
}
lpszToken = FindOneOf(lpszToken, szTokens);
}
return true;
} BOOL CServicesModule::Install() throw()
{
if (IsInstalled())
return TRUE; // Get the executable file path
TCHAR szFilePath[MAX_PATH + _ATL_QUOTES_SPACE];
DWORD dwFLen = ::GetModuleFileName(NULL, szFilePath + 1, MAX_PATH);
if( dwFLen == 0 || dwFLen == MAX_PATH )
return FALSE; // Quote the FilePath before calling CreateService
szFilePath[0] = _T('\"');
szFilePath[dwFLen + 1] = _T('\"');
szFilePath[dwFLen + 2] = 0; SC_HANDLE hSCM = ::OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS); //创建服务、根据命令行设置启动方式,设置依赖关系
SC_HANDLE hService = ::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
dwStartType, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, dependServices, NULL, NULL); if (hService == NULL)
{
::CloseServiceHandle(hSCM);
return FALSE;
} ::CloseServiceHandle(hService);
::CloseServiceHandle(hSCM);
return TRUE;
}

 

自定义命令行参数演示

注册服务时使用如下命令行

Services.exe -Auto -service CryptSvc RPCSS DcomLaunch

注册后,效果如下

 

系列链接

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

玩转Windows服务系列——Debug、Release版本的注册和卸载,及其原理

玩转Windows服务系列——无COM接口Windows服务启动失败原因及解决方案

玩转Windows服务系列——服务运行、停止流程浅析

玩转Windows服务系列——Windows服务小技巧

玩转Windows服务系列——命令行管理Windows服务

玩转Windows服务系列——Windows服务小技巧的更多相关文章

  1. 玩转Windows服务系列——Windows服务小技巧

    伴随着研究Windows服务,逐渐掌握了一些小技巧,现在与大家分享一下. 将Windows服务转变为控制台程序 由于默认的Windows服务程序,编译后为Win32的窗口程序.我们在程序启动或运行过程 ...

  2. 玩转Windows服务系列——Windows服务启动超时时间

    最近有客户反映,机房出现断电情况,服务器的系统重新启动后,数据库服务自启动失败.第一次遇到这种情况,为了查看是不是断电情况导致数据库文件损坏,从客户的服务器拿到数据库的日志,进行分析. 数据库工作机制 ...

  3. go微服务系列(二) - 服务注册/服务发现

    目录 1. 服务注册 1.1 代码演示 1.2 在go run的时候传入服务注册的参数 2. 服务发现均衡负载 2.1 均衡负载算法 2.2 服务发现均衡负载的演示 1. 服务注册 1.1 代码演示 ...

  4. go微服务系列(三) - 服务调用(http)

    1. 关于服务调用 2. 基本方式调用服务 3. 服务调用正确姿势(初步) 3.1 服务端代码 3.2 客户端调用(重要) 1. 关于服务调用 这里的服务调用,我们调用的可以是http api也可以是 ...

  5. 玩转Windows服务系列——给Windows服务添加COM接口

    当我们运行一个Windows服务的时候,一般情况下,我们会选择以非窗口或者非控制台的方式运行,这样,它就只是一个后台程序,没有界面供我们进行交互. 那么当我们想与Windows服务进行实时交互的时候, ...

  6. 玩转Windows服务系列——使用Boost.Application快速构建Windows服务

    玩转Windows服务系列——创建Windows服务一文中,介绍了如何快速使用VS构建一个Windows服务.Debug.Release版本的注册和卸载,及其原理和服务运行.停止流程浅析分别介绍了Wi ...

  7. 玩转Windows服务系列——Debug、Release版本的注册和卸载,及其原理

    Windows服务Debug版本 注册 Services.exe -regserver 卸载 Services.exe -unregserver Windows服务Release版本 注册 Servi ...

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

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

  9. 玩转Windows服务系列——无COM接口Windows服务启动失败原因及解决方案

    将VS创建的Windows服务项目编译生成的程序,通过命令行 “服务.exe -Service”注册为Windows服务后,就可以通过服务管理器进行管理了. 问题 通过服务管理器进行启动的时候,发现服 ...

随机推荐

  1. 【HTML5游戏开发小技巧】RPG情景对话中,令文本逐字输出

    以前用JAVAscript实现过令文本逐字输出的效果,今天我来用html5中的canvas实现一下.canvas里的内容可不像<p>那样好操作,首先,你需要懂得一些html5的API才能操 ...

  2. 第十一章 认识与学习BASH

    系统支持的shell在   /etc/shells里面 Bash Shell 的功能: 1.命令修补能力(histroy) 2.命令与档案补全功能 3.命令别名设定功能 4.工作前景背景控制 5.支持 ...

  3. js和循环依赖

    kmdjs和循环依赖 循环依赖是非常必要的,有的程序写着写着就循环依赖了,可以提取出一个对象来共同依赖解决循环依赖,但是有时会破坏程序的逻辑自封闭和高内聚.所以没解决好循环依赖的模块化库.框架.编译器 ...

  4. HTTP 错误 500.19 – Internal Server Error web.config 文件的 system.webServer/httpErrors 节中不允许绝对物理路径“C:\inetpub\custerr”[转]

    给ASP或者ASP.NET等需要配置IIS服务器的过程中,很可能会遇到以下两种错误.尤其是用Win7系统的,配置IIS7.0版本比用XP系统配置IIS5.1版本而言要复杂复杂一些.当同时需要配置ASP ...

  5. anglehack参赛总结

    自已不足的方面: 自已和伙伴是带着idea去的,但是没有带有很大的热情激励和吸引在场的hacker加入团队,一定要找最优秀的人加入团队,事实上我是有这方面识人认人的能力的,24h,5-6个人的优秀团队 ...

  6. java假设模拟请求重新启动路由器(网络爬虫经常使用),还有java怎样下载图片

    我们假设在公司或家里使用网络爬虫去抓取自己索要的一些数据的时候,经常对方的站点有defence机制,会给你的http请求返回500错误,仅仅要是同样IP就请求不到数据,这时候我们仅仅能去重新启动路由器 ...

  7. 杂题_POJ上的过桥问题

    本文出自:http://blog.csdn.net/svitter 过桥问题解释:一条船能够坐两个人,可是有非常多人要过河,所以送过一个人去,还有一个人还要回来接.使全部人过河之后时间最短,怎样求? ...

  8. PHP中遍历stdclass object 及 json 总结[中国航天神舟十号以json形式向地面返回数据]

    $test=Array ( [0] => stdClass Object ( [tags] => 最快的车,Bloodhound,SSC [id] => 48326888 11 从网 ...

  9. 浮点数在计算机内存中的表示(IEEE 754规定1位是符号位,8位是指数,剩下的23位为有效数字)

    本文转载自:阮一峰的博客,http://www.ruanyifeng.com/blog/2010/06/ieee_floating-point_representation.html 张玉彬的博客 h ...

  10. AssertValid函数学�

    转自http://tsitao.blog.163.com/blog/static/29795822006914105840496/ VC的调试中,AssertValid和Dump函数的应用 CObje ...