下面介绍使用VC++开发Windows服务程序。
首先运行VC++6,选择新建工程,在出现的下面窗口中选择”ATL COM AppWizard“,并选择工程放置位置和相应的工程名,然后选择确定按钮。
此时出现如下图窗口界面,在此界面中选择”服务(EXE)“,然后选择完成按钮。
在接下来的窗口中选择确定按钮。
则VC完成向导并生成相应代码(效果如下)。
程序的进入点是全局函数_tWinMain, 仔细看一下这个函数,我们会发现当我们运行程序时,可以加上参数,例如: winsvr /RegServer 或者 winsvr -RegServer,这个是用来本地服务器注册(Register as Local S Register as Service erver)。
其中winsvr / Service 或者 winsvr -Service,这个是服务的注册(Register as Service);winsvr /UnRegServer 或者 winsvr -UnRegServer ,这个是服务的删除。
所以,当我们写好了服务程序,只要运行的时候加上参数 Service ,这个时候在SCM中就会看到我们的服务了。
每次编码后测试都要在命令行中加参数运行服务才可以在SCM中列出来这样很麻烦,因此可以采用如下方式来处理:选择VC IDE的菜单工程 -> 设置, 再选择自定义组建面板(如下图所示)
在"$(TargetPath)" /RegServer的下面加上:"$(TargetPath)" /Service,这样当我们每次编码后编译程序,就不用再在命令行中去加参数执行我们的服务程序完成服务的注册了。
同时通过界面我们也看到,向导为我们建立了一个类:CServiceModule,全局变量_Module就是这个类的实例。
Init():这个函数用于完成一些初始化工作;
Run():这个函数就是服务开始运行后的内容,我们接下来要修改的内容也就是从这里入手。
Install():有如下一段代码
SC_HANDLE hService = ::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0"), NULL, NULL);
注意:如果服务中启动的程序具有窗口(即具有交互功能则要求使用如下代码)
SC_HANDLE hService = ::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_SHARE_PROCESS | SERVICE_INTERACTIVE_PROCESS,
SERVICE_AUTO_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0"), NULL, NULL);
这个CreateService函数原先如下:
SC_HANDLE CreateService(
SC_HANDLE hSCManager, // handle to SCM database
LPCTSTR lpServiceName, // name of service to start
LPCTSTR lpDisplayName, // display name
DWORD dwDesiredAccess, // type of access to service
DWORD dwServiceType, // type of service
DWORD dwStartType, // when to start service
DWORD dwErrorControl, // severity of service failure
LPCTSTR lpBinaryPathName, // name of binary file
LPCTSTR lpLoadOrderGroup, // name of load ordering group
LPDWORD lpdwTagId, // tag identifier
LPCTSTR lpDependencies, // array of dependency names
LPCTSTR lpServiceStartName, // account name
LPCTSTR lpPassword // account password
);
第六个参数是服务的启动类型。
SERVICE_DEMAND_START是手动启动,SERVICE_AUTO_START是自动启动。
第十一个参数是服务的依存关系,比如说服务的启动想要依存SQL Server的启动,那我们可以把这个参数写成:
_T("MSSQLSERVER\0");
如果我们写的服务不依存于其他的任何服务,那我们就将此参数设置为NULL就可以了。
接下来我们实现我们需要实现的业务。
首先,我们在类CServiceModule中找到Run函数,并在Run函数中找到以下代码:
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
并在此代码前加入自己的代码,我这里加入的代码:CustomFunc1()。
这里对应不同应用有不同写法:
1、如果要启动一个窗口进行交互,则代码如下:
定义两个成员函数CustomFunc1和CustomFunc2
void CServiceModule::CustomFunc1()
{
TCHAR szFilePath[MAX_PATH + 1];
GetModuleFileName(NULL, szFilePath, MAX_PATH);
(_tcsrchr(szFilePath, _T('\\')))[1] = 0; //删除文件名,只获得路径
CString str_url = szFilePath;
str_url=str_url+"B.exe";
LogEvent("运行程序:"+str_url);
UINT rtn=WinExec(str_url,SW_SHOW );
//UINT rtn=WinExec("c:\\windows\\system32\\cmd.exe",SW_SHOWNORMAL);
if(rtn>31){
LogEvent("调用成功");
}
else{
if(rtn==0){
LogEvent("内存不足");
}
else{
if(rtn==ERROR_BAD_FORMAT){ //ERROR_BAD_FORMAT = 11
LogEvent("EXE 文件无效");
}
else{
if(rtn==ERROR_FILE_NOT_FOUND){ //ERROR_FILE_NOT_FOUND = 2
LogEvent("文件名错误");
}
else{
if(rtn==ERROR_PATH_NOT_FOUND){ //ERROR_PATH_NOT_FOUND = 3
LogEvent("路径名错误");
}
else{
LogEvent("发生其它错误");
}
}
}
}
}
}
BOOL CServiceModule::CustomFunc2()
{
HDESK hdeskCurrent;
HDESK hdesk;
HWINSTA hwinstaCurrent;
HWINSTA hwinsta;
hwinstaCurrent = GetProcessWindowStation();
if (hwinstaCurrent == NULL){
LogEvent(_T("get window station err"));
return FALSE;
}
hdeskCurrent = GetThreadDesktop(GetCurrentThreadId());
if (hdeskCurrent == NULL){
LogEvent(_T("get window desktop err"));
return FALSE;
}
//打开winsta0
hwinsta = OpenWindowStation("winsta0", FALSE,
WINSTA_ACCESSCLIPBOARD |
WINSTA_ACCESSGLOBALATOMS |
WINSTA_CREATEDESKTOP |
WINSTA_ENUMDESKTOPS |
WINSTA_ENUMERATE |
WINSTA_EXITWINDOWS |
WINSTA_READATTRIBUTES |
WINSTA_READSCREEN |
WINSTA_WRITEATTRIBUTES);
if (hwinsta == NULL){
LogEvent(_T("open window station err"));
return FALSE;
}
if (!SetProcessWindowStation(hwinsta)){
LogEvent(_T("Set window station err"));
return FALSE;
}
//打开desktop
hdesk = OpenDesktop("default", 0, FALSE,
DESKTOP_CREATEMENU |
DESKTOP_CREATEWINDOW |
DESKTOP_ENUMERATE |
DESKTOP_HOOKCONTROL |
DESKTOP_JOURNALPLAYBACK |
DESKTOP_JOURNALRECORD |
DESKTOP_READOBJECTS |
DESKTOP_SWITCHDESKTOP |
DESKTOP_WRITEOBJECTS);
if (hdesk == NULL){
LogEvent(_T("Open desktop err"));
return FALSE;
}
SetThreadDesktop(hdesk);
CustomFunc1();
if (!SetProcessWindowStation(hwinstaCurrent))
return FALSE;
if (!SetThreadDesktop(hdeskCurrent))
return FALSE;
if (!CloseWindowStation(hwinsta))
return FALSE;
if (!CloseDesktop(hdesk))
return FALSE;
return TRUE;
}
2、无交互窗口,则代码如下:
void CServiceModule::CustomFunc1(){
LogEvent(_T("Custorm Function Invoked")); //这里可以改成任何你需要的代码,当然不能有显示窗口之内的代码,要显示窗口之内代码请采用上面方法1。
}
现在可以编译并执行程序了。
此时会编译时会报告一个错误:'CString' : undeclared identifier。
这时需要查看工程的一些设置:
菜单工程->设置 ,常规面板,默认的设置是:使用MFC作为静态连接库。如果是这个设置则做如下工作:
然后我们打开StdAfx.h文件,并找到#include <atlbase.h>位置,并在它之前加入#include <afx.h>。重新编译可以了。
如果需要修改出现在scm中的服务名,可以在工程中找到资源文件中的IDS_SERVICENAME项的内容就可以了。
注意:编译好了的程序需要在控制台中输入如下指令
1、注册服务
winsvr /regserver
winsvr /service //此条指令十分重要,如果不执行则在scm中是看不到此服务的
2、注销服务
winsvr /unregserver
首先运行VC++6,选择新建工程,在出现的下面窗口中选择”ATL COM AppWizard“,并选择工程放置位置和相应的工程名,然后选择确定按钮。
此时出现如下图窗口界面,在此界面中选择”服务(EXE)“,然后选择完成按钮。
在接下来的窗口中选择确定按钮。
则VC完成向导并生成相应代码(效果如下)。
程序的进入点是全局函数_tWinMain, 仔细看一下这个函数,我们会发现当我们运行程序时,可以加上参数,例如: winsvr /RegServer 或者 winsvr -RegServer,这个是用来本地服务器注册(Register as Local S Register as Service erver)。
其中winsvr / Service 或者 winsvr -Service,这个是服务的注册(Register as Service);winsvr /UnRegServer 或者 winsvr -UnRegServer ,这个是服务的删除。
所以,当我们写好了服务程序,只要运行的时候加上参数 Service ,这个时候在SCM中就会看到我们的服务了。
每次编码后测试都要在命令行中加参数运行服务才可以在SCM中列出来这样很麻烦,因此可以采用如下方式来处理:选择VC IDE的菜单工程 -> 设置, 再选择自定义组建面板(如下图所示)
在"$(TargetPath)" /RegServer的下面加上:"$(TargetPath)" /Service,这样当我们每次编码后编译程序,就不用再在命令行中去加参数执行我们的服务程序完成服务的注册了。
同时通过界面我们也看到,向导为我们建立了一个类:CServiceModule,全局变量_Module就是这个类的实例。
Init():这个函数用于完成一些初始化工作;
Run():这个函数就是服务开始运行后的内容,我们接下来要修改的内容也就是从这里入手。
Install():有如下一段代码
SC_HANDLE hService = ::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS,
SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0"), NULL, NULL);
注意:如果服务中启动的程序具有窗口(即具有交互功能则要求使用如下代码)
SC_HANDLE hService = ::CreateService(
hSCM, m_szServiceName, m_szServiceName,
SERVICE_ALL_ACCESS, SERVICE_WIN32_SHARE_PROCESS | SERVICE_INTERACTIVE_PROCESS,
SERVICE_AUTO_START, SERVICE_ERROR_NORMAL,
szFilePath, NULL, NULL, _T("RPCSS\0"), NULL, NULL);
这个CreateService函数原先如下:
SC_HANDLE CreateService(
SC_HANDLE hSCManager, // handle to SCM database
LPCTSTR lpServiceName, // name of service to start
LPCTSTR lpDisplayName, // display name
DWORD dwDesiredAccess, // type of access to service
DWORD dwServiceType, // type of service
DWORD dwStartType, // when to start service
DWORD dwErrorControl, // severity of service failure
LPCTSTR lpBinaryPathName, // name of binary file
LPCTSTR lpLoadOrderGroup, // name of load ordering group
LPDWORD lpdwTagId, // tag identifier
LPCTSTR lpDependencies, // array of dependency names
LPCTSTR lpServiceStartName, // account name
LPCTSTR lpPassword // account password
);
第六个参数是服务的启动类型。
SERVICE_DEMAND_START是手动启动,SERVICE_AUTO_START是自动启动。
第十一个参数是服务的依存关系,比如说服务的启动想要依存SQL Server的启动,那我们可以把这个参数写成:
_T("MSSQLSERVER\0");
如果我们写的服务不依存于其他的任何服务,那我们就将此参数设置为NULL就可以了。
接下来我们实现我们需要实现的业务。
首先,我们在类CServiceModule中找到Run函数,并在Run函数中找到以下代码:
MSG msg;
while (GetMessage(&msg, 0, 0, 0))
DispatchMessage(&msg);
并在此代码前加入自己的代码,我这里加入的代码:CustomFunc1()。
这里对应不同应用有不同写法:
1、如果要启动一个窗口进行交互,则代码如下:
定义两个成员函数CustomFunc1和CustomFunc2
void CServiceModule::CustomFunc1()
{
TCHAR szFilePath[MAX_PATH + 1];
GetModuleFileName(NULL, szFilePath, MAX_PATH);
(_tcsrchr(szFilePath, _T('\\')))[1] = 0; //删除文件名,只获得路径
CString str_url = szFilePath;
str_url=str_url+"B.exe";
LogEvent("运行程序:"+str_url);
UINT rtn=WinExec(str_url,SW_SHOW );
//UINT rtn=WinExec("c:\\windows\\system32\\cmd.exe",SW_SHOWNORMAL);
if(rtn>31){
LogEvent("调用成功");
}
else{
if(rtn==0){
LogEvent("内存不足");
}
else{
if(rtn==ERROR_BAD_FORMAT){ //ERROR_BAD_FORMAT = 11
LogEvent("EXE 文件无效");
}
else{
if(rtn==ERROR_FILE_NOT_FOUND){ //ERROR_FILE_NOT_FOUND = 2
LogEvent("文件名错误");
}
else{
if(rtn==ERROR_PATH_NOT_FOUND){ //ERROR_PATH_NOT_FOUND = 3
LogEvent("路径名错误");
}
else{
LogEvent("发生其它错误");
}
}
}
}
}
}
BOOL CServiceModule::CustomFunc2()
{
HDESK hdeskCurrent;
HDESK hdesk;
HWINSTA hwinstaCurrent;
HWINSTA hwinsta;
hwinstaCurrent = GetProcessWindowStation();
if (hwinstaCurrent == NULL){
LogEvent(_T("get window station err"));
return FALSE;
}
hdeskCurrent = GetThreadDesktop(GetCurrentThreadId());
if (hdeskCurrent == NULL){
LogEvent(_T("get window desktop err"));
return FALSE;
}
//打开winsta0
hwinsta = OpenWindowStation("winsta0", FALSE,
WINSTA_ACCESSCLIPBOARD |
WINSTA_ACCESSGLOBALATOMS |
WINSTA_CREATEDESKTOP |
WINSTA_ENUMDESKTOPS |
WINSTA_ENUMERATE |
WINSTA_EXITWINDOWS |
WINSTA_READATTRIBUTES |
WINSTA_READSCREEN |
WINSTA_WRITEATTRIBUTES);
if (hwinsta == NULL){
LogEvent(_T("open window station err"));
return FALSE;
}
if (!SetProcessWindowStation(hwinsta)){
LogEvent(_T("Set window station err"));
return FALSE;
}
//打开desktop
hdesk = OpenDesktop("default", 0, FALSE,
DESKTOP_CREATEMENU |
DESKTOP_CREATEWINDOW |
DESKTOP_ENUMERATE |
DESKTOP_HOOKCONTROL |
DESKTOP_JOURNALPLAYBACK |
DESKTOP_JOURNALRECORD |
DESKTOP_READOBJECTS |
DESKTOP_SWITCHDESKTOP |
DESKTOP_WRITEOBJECTS);
if (hdesk == NULL){
LogEvent(_T("Open desktop err"));
return FALSE;
}
SetThreadDesktop(hdesk);
CustomFunc1();
if (!SetProcessWindowStation(hwinstaCurrent))
return FALSE;
if (!SetThreadDesktop(hdeskCurrent))
return FALSE;
if (!CloseWindowStation(hwinsta))
return FALSE;
if (!CloseDesktop(hdesk))
return FALSE;
return TRUE;
}
2、无交互窗口,则代码如下:
void CServiceModule::CustomFunc1(){
LogEvent(_T("Custorm Function Invoked")); //这里可以改成任何你需要的代码,当然不能有显示窗口之内的代码,要显示窗口之内代码请采用上面方法1。
}
现在可以编译并执行程序了。
此时会编译时会报告一个错误:'CString' : undeclared identifier。
这时需要查看工程的一些设置:
菜单工程->设置 ,常规面板,默认的设置是:使用MFC作为静态连接库。如果是这个设置则做如下工作:
然后我们打开StdAfx.h文件,并找到#include <atlbase.h>位置,并在它之前加入#include <afx.h>。重新编译可以了。
如果需要修改出现在scm中的服务名,可以在工程中找到资源文件中的IDS_SERVICENAME项的内容就可以了。
注意:编译好了的程序需要在控制台中输入如下指令
1、注册服务
winsvr /regserver
winsvr /service //此条指令十分重要,如果不执行则在scm中是看不到此服务的
2、注销服务
winsvr /unregserver