C/C++ 開發SCM服務管理組件

来源:https://www.cnblogs.com/LyShark/archive/2023/11/23/17850974.html
-Advertisement-
Play Games

SCM(Service Control Manager)服務管理器是 Windows 操作系統中的一個關鍵組件,負責管理系統服務的啟動、停止和配置。服務是一種在後臺運行的應用程式,可以在系統啟動時自動啟動,也可以由用戶或其他應用程式手動啟動。本篇文章中,我們將通過使用 Windows 的服務管理器(... ...


SCM(Service Control Manager)服務管理器是 Windows 操作系統中的一個關鍵組件,負責管理系統服務的啟動、停止和配置。服務是一種在後臺運行的應用程式,可以在系統啟動時自動啟動,也可以由用戶或其他應用程式手動啟動。本篇文章中,我們將通過使用 Windows 的服務管理器(SCM)提供的API介面,實現一個簡單的服務管理組件的編寫。

服務管理器的主要功能包括:

  1. 服務啟動和停止: SCM 管理系統服務的啟動和停止。在系統啟動時,SCM 會根據每個服務的配置啟動相應的服務。用戶也可以通過服務管理器手動啟動或停止服務。
  2. 服務配置: SCM 管理服務的配置信息,包括服務的啟動類型(如自動、手動、禁用)、服務的依賴關係、服務的用戶身份等。
  3. 服務狀態監控: SCM 監控運行中服務的狀態。服務可以處於運行、暫停、停止等狀態。SCM 提供 API 函數,允許應用程式查詢和控制服務的狀態。
  4. 事件日誌: SCM 記錄服務啟動、停止等事件到系統的事件日誌中,這有助於故障排查和系統管理。
  5. 服務通知: SCM 允許應用程式註冊服務狀態變化的通知,以便及時響應服務狀態的改變。
  6. 服務安全性: SCM 確保服務以適當的許可權和身份運行,以保障系統的安全性。

開發者可以通過使用 Windows API 提供的相關函數(例如 OpenSCManagerCreateServiceStartService 等)與 SCM 進行交互,管理系統中的服務。這些 API 函數允許開發者創建、配置、啟動、停止和查詢服務,以及監控服務的狀態變化。

枚舉SCM系統服務

Windows 的服務控制管理器(SCM)允許開發者通過 EnumServicesStatus 函數來枚舉系統中正在運行的服務。這個功能非常有用,可以用於監控系統中的服務狀態、獲取服務的詳細信息等。在這篇文章中,我們將學習如何使用 EnumServicesStatus 函數來實現對 SCM 系統服務的枚舉,並獲取相關信息。

OpenSCManager 用於打開服務控制管理器資料庫,並返回一個指向服務控制管理器的句柄。通過這個句柄,你可以進行對服務的查詢、創建、啟動、停止等操作。

以下是 OpenSCManager 函數的原型:

SC_HANDLE OpenSCManager(
  LPCTSTR lpMachineName,
  LPCTSTR lpDatabaseName,
  DWORD   dwDesiredAccess
);
  • lpMachineName: 指定遠程電腦的名稱。如果為 NULL,表示本地電腦。

  • lpDatabaseName: 指定要打開的服務控制管理器資料庫的名稱。通常為 SERVICES_ACTIVE_DATABASE

  • dwDesiredAccess
    

    : 指定所請求的訪問許可權。可以是以下之一或它們的組合:

    • SC_MANAGER_CONNECT: 允許連接服務控制管理器。
    • SC_MANAGER_CREATE_SERVICE: 允許創建服務。
    • SC_MANAGER_ENUMERATE_SERVICE: 允許枚舉服務。
    • SC_MANAGER_LOCK: 允許鎖定服務資料庫。
    • SC_MANAGER_QUERY_LOCK_STATUS: 允許查詢服務資料庫的鎖定狀態。
    • SC_MANAGER_MODIFY_BOOT_CONFIG: 允許修改系統啟動配置。
    • SC_MANAGER_ALL_ACCESS: 允許執行上述所有操作。

函數返回一個指向服務控制管理器的句柄 (SC_HANDLE)。如果操作失敗,返回 NULL,可以通過調用 GetLastError 函數獲取錯誤代碼。

EnumServicesStatus 用於枚舉指定服務控制管理器資料庫中的服務。通過這個函數,你可以獲取正在運行的服務的信息,如服務的名稱、顯示名稱、狀態等。

以下是 EnumServicesStatus 函數的原型:

BOOL EnumServicesStatus(
  SC_HANDLE hSCManager,
  DWORD     dwServiceType,
  DWORD     dwServiceState,
  LPENUM_SERVICE_STATUS lpServices,
  DWORD     cbBufSize,
  LPDWORD   pcbBytesNeeded,
  LPDWORD   lpServicesReturned,
  LPDWORD   lpResumeHandle
);
  • hSCManager: 指定服務控制管理器的句柄,通過 OpenSCManager 函數獲取。
  • dwServiceType: 指定服務的類型,如 SERVICE_WIN32
  • dwServiceState: 指定服務的狀態,如 SERVICE_STATE_ALL
  • lpServices: 指向 ENUM_SERVICE_STATUS 結構體數組的指針,用於接收服務的信息。
  • cbBufSize: 指定 lpServices 緩衝區的大小,以位元組為單位。
  • pcbBytesNeeded: 接收所需的緩衝區大小,以位元組為單位。
  • lpServicesReturned: 接收實際返回的服務數。
  • lpResumeHandle: 用於標識服務的遍歷位置。

函數返回 BOOL 類型,如果調用成功,返回 TRUE,否則返回 FALSE。如果函數返回 FALSE,可以通過調用 GetLastError 函數獲取錯誤代碼。

上述EnumServicesStatus中的第二個參數dwServiceType非常重要,在 Windows 操作系統中,服務的啟動類型和服務類型是通過服務的標誌(Service Flags)來指定的。這些標誌是用於定義服務的性質和啟動方式的。以下是其中幾個標誌的含義:

  1. 0x0 (SERVICE_KERNEL_DRIVER): 設備驅動程式。這種服務類型表示一個內核模式的設備驅動程式。
  2. 0x2 (SERVICE_FILE_SYSTEM_DRIVER): 內核模式文件系統驅動程式。這種服務類型表示一個內核模式的文件系統驅動程式。
  3. 0x8 (SERVICE_FILE_SYSTEM_DRIVER | SERVICE_RECOGNIZER_DRIVER): 文件系統識別器驅動程式。這種服務類型表示一個同時具有文件系統驅動程式和文件系統識別器驅動程式功能的服務。
  4. 0x10 (SERVICE_WIN32_OWN_PROCESS): 獨占一個進程的服務。這種服務類型表示服務運行在自己的進程中。
  5. 0x20 (SERVICE_WIN32_SHARE_PROCESS): 與其他服務共用一個進程的服務。這種服務類型表示服務可以與其他服務運行在同一個進程中。

需要註意的是,上述標誌可以通過按位 OR 運算來組合使用,以表示服務的多個特性。例如,SERVICE_WIN32_OWN_PROCESS | SERVICE_INTERACTIVE_PROCESS 表示一個互動式服務,即運行在自己的進程中並與桌面交互。

除了上述標誌之外,還有一些其他的標誌,如:

  • SERVICE_INTERACTIVE_PROCESS (0x100): 互動式服務。表示服務可以與桌面進行交互,通常用於服務需要顯示用戶界面的情況。
  • SERVICE_AUTO_START (0x2): 自動啟動。表示服務會在系統啟動時自動啟動。
  • SERVICE_DEMAND_START (0x3): 手動啟動。表示服務需要由用戶手動啟動。
  • SERVICE_DISABLED (0x4): 禁用。表示服務被禁用,不會自動啟動。

這些標誌允許開發者靈活地定義服務的啟動方式和性質。在使用服務相關的 API 函數時,這些標誌會在函數參數中進行指定。例如,在使用 CreateService 函數時,可以通過設置 dwServiceTypedwStartType 參數來指定服務的類型和啟動方式。

如下代碼則實現了對系統內特定服務的枚舉功能,通過向Enum_Services函數中傳入不同的參數來實現枚舉不同的服務類型;

#include <stdio.h>
#include <Windows.h>

void Enum_Services(DWORD dwServiceType)
{
	DWORD ServiceCount = 0, dwSize = 0;
	LPENUM_SERVICE_STATUS lpInfo;

	SC_HANDLE hSCM = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
	BOOL bRet = EnumServicesStatus(hSCM, dwServiceType, SERVICE_STATE_ALL, NULL, 0, &dwSize, &ServiceCount, NULL);
	if (!bRet && GetLastError() == ERROR_MORE_DATA)
	{
		// 分配緩衝區,保存服務列表
		lpInfo = (LPENUM_SERVICE_STATUS)(new BYTE[dwSize]);
		bRet = EnumServicesStatus(hSCM, dwServiceType, SERVICE_STATE_ALL, (LPENUM_SERVICE_STATUS)lpInfo,
			dwSize, &dwSize, &ServiceCount, NULL);
		
		if (NULL == hSCM)
		{
			return;
		}

		// 逐個遍歷獲取服務信息
		for (int x = 0; x < ServiceCount; x++)
		{
			printf("名稱:%-30s 名稱: %-30s 狀態: ", lpInfo[x].lpServiceName, lpInfo[x].lpDisplayName);
			switch (lpInfo[x].ServiceStatus.dwCurrentState)
			{
			case SERVICE_PAUSED:  printf("暫停 \n"); break;
			case SERVICE_STOPPED: printf("停止 \n"); break;
			case SERVICE_RUNNING: printf("運行 (*) \n"); break;
			default: printf("其他 \n");
			}
		}
		delete lpInfo;
	}
	CloseServiceHandle(hSCM);
}

int main(int argc, char *argv[])
{
	// 0x0 => 設備驅動程式
	// 0x2=> 內核模式文件系統驅動程式
	// 0x8 => 文件系統識別器驅動程式
	// 0x10 => 獨占一個進程的服務
	// 0x20 => 與其他服務共用一個進程的服務

	Enum_Services(0x10);
	system("pause");
	return 0;
}

我們傳入0x10則代表枚舉當前系統中的獨占一個進程的服務,代碼需要使用管理員許可權運行,輸出效果圖如下所示;

編寫SCM系統服務

Windows 服務程式的主體框架需要包括關鍵的兩個函數,其中ServiceMain標志著服務程式的入口,而ServiceCtrlHandle則是服務程式的控制處理流程,最後的TellSCM函數則用於通知SCM服務的當前狀態,當然了TellSCM可以單獨出來也可以寫在ServiceCtrlHandle都可以,任何一個正常的服務程式都必須包含這兩個關鍵位置,並且需要將該函數導出,首先展示核心API函數的定義信息。

SERVICE_TABLE_ENTRY 用於定義服務表的結構體。服務表是一個包含服務入口函數和服務名的數組,它告訴 SCM (服務控制管理器)哪個服務程式入口函數與哪個服務相關聯。

以下是 SERVICE_TABLE_ENTRY 結構體的定義:

typedef struct _SERVICE_TABLE_ENTRY {
    LPSTR lpServiceName;          // 服務名
    LPSERVICE_MAIN_FUNCTION lpServiceProc;  // 服務入口函數
} SERVICE_TABLE_ENTRY, *LPSERVICE_TABLE_ENTRY;
  • lpServiceName: 指向服務名的指針。服務名是服務在 SCM 中的標識符,可以通過該名字啟動、停止、控制服務等。
  • lpServiceProc: 指向服務入口函數的指針。該函數是服務的主要執行點,當 SCM 啟動服務時會調用該函數。

在主程式中,你通過創建 SERVICE_TABLE_ENTRY 數組來定義服務表,然後將其傳遞給 StartServiceCtrlDispatcher 函數。代碼中,服務表包含一個 SERVICE_TABLE_ENTRY 結構體:

SERVICE_TABLE_ENTRY stDispatchTable[] = {
    { g_szServiceName, (LPSERVICE_MAIN_FUNCTION)ServiceMain },
    { NULL, NULL }
};
  • g_szServiceName: 是你的服務的名字,這裡定義了為 "ServiceTest.exe"。
  • (LPSERVICE_MAIN_FUNCTION)ServiceMain: 是指向服務入口函數 ServiceMain 的指針。當 SCM 啟動服務時,將調用這個函數。

這個服務表告訴 SCM 與哪個服務相關聯,通過哪個函數來啟動和管理服務。 StartServiceCtrlDispatcher 函數接受這個服務表作為參數,並負責將控制傳遞給適當的服務。

StartServiceCtrlDispatcher 用於啟動服務控制分發器。這個函數通常在服務程式的 main 函數中調用,它接受一個包含服務表的數組作為參數,並將控制傳遞給適當的服務。

以下是 StartServiceCtrlDispatcher 函數的原型:

BOOL StartServiceCtrlDispatcher(
  const SERVICE_TABLE_ENTRY *lpServiceTable
);
  • lpServiceTable: 指向 SERVICE_TABLE_ENTRY 結構體數組的指針,該數組定義了服務表。服務表中的每個元素指定了服務的名稱和服務入口函數。

該函數返回 BOOL 類型。如果調用成功,返回 TRUE,否則返回 FALSE。如果返回 FALSE,可以通過調用 GetLastError 函數獲取錯誤代碼。

RegisterServiceCtrlHandler 用於註冊一個服務控制處理程式,該處理程式將接收來自 SCM(服務控制管理器)的控制請求。每個服務都需要註冊一個服務控制處理程式,以便在服務狀態發生變化時接收通知。

以下是 RegisterServiceCtrlHandler 函數的原型:

SERVICE_STATUS_HANDLE RegisterServiceCtrlHandler(
  LPCTSTR                  lpServiceName,
  LPHANDLER_FUNCTION_EX    lpHandlerProc
);
  • lpServiceName: 指定要註冊的服務的名稱。這應該是服務在 SCM 中註冊的唯一標識符。
  • lpHandlerProc: 指定服務控制處理程式的地址。這是一個指向處理函數的指針,該函數將在接收到控制請求時被調用。

函數返回一個 SERVICE_STATUS_HANDLE 類型的句柄。這個句柄用於標識服務控制管理器中的服務控制處理程式。

SetServiceStatus 用於通知 SCM(服務控制管理器)關於服務的當前狀態。這個函數通常在服務的主迴圈中調用,以便及時向 SCM 報告服務的狀態變化。

以下是 SetServiceStatus 函數的原型:

BOOL SetServiceStatus(
  SERVICE_STATUS_HANDLE hServiceStatus,
  LPSERVICE_STATUS      lpServiceStatus
);
  • hServiceStatus: 指定服務控制管理器中的服務的句柄,即由 RegisterServiceCtrlHandler 返回的句柄。
  • lpServiceStatus: 指向 SERVICE_STATUS 結構體的指針,該結構體描述了服務的當前狀態。

SERVICE_STATUS 結構體定義如下:

typedef struct _SERVICE_STATUS {
  DWORD dwServiceType;
  DWORD dwCurrentState;
  DWORD dwControlsAccepted;
  DWORD dwWin32ExitCode;
  DWORD dwServiceSpecificExitCode;
  DWORD dwCheckPoint;
  DWORD dwWaitHint;
} SERVICE_STATUS, *LPSERVICE_STATUS;
  • dwServiceType: 服務的類型,例如 SERVICE_WIN32_OWN_PROCESS
  • dwCurrentState: 服務的當前狀態,例如 SERVICE_RUNNING
  • dwControlsAccepted: 服務接受的控制碼,例如 SERVICE_ACCEPT_STOP 表示服務接受停止控制。
  • dwWin32ExitCode: 服務的 Win32 退出碼。
  • dwServiceSpecificExitCode: 服務的特定退出碼。
  • dwCheckPoint: 在操作進行中時,用於指示操作的進度。
  • dwWaitHint: SCM 期望服務完成操作所需的等待時間。

有了上述介面的說明,並通過遵循微軟的對服務編寫的定義即可實現一個系統服務,這裡的DoTask()是一個自定義函數,該服務在啟動後會率先執行此處,此處可用於定義特定的功能,例如開機自啟動某個進程,或者是遠程創建套接字等,當然了服務程式也可以是exe如下可以使用控制台方式創建。

#include <Windows.h>

// 服務入口函數以及處理回調函數
void __stdcall ServiceMain(DWORD dwArgc, char *lpszArgv);
void __stdcall ServiceCtrlHandle(DWORD dwOperateCode);
BOOL TellSCM(DWORD dwState, DWORD dwExitCode, DWORD dwProgress);
// 自定義函數
void DoTask();

// 全局變數
char g_szServiceName[MAX_PATH] = "ServiceTest.exe";    // 自身服務名稱 
SERVICE_STATUS_HANDLE g_ServiceStatusHandle = { 0 };

int main(int argc, char * argv[])
{
  // 註冊服務入口函數
  SERVICE_TABLE_ENTRY stDispatchTable[] = { { g_szServiceName, (LPSERVICE_MAIN_FUNCTION)ServiceMain }, { NULL, NULL } };
  ::StartServiceCtrlDispatcher(stDispatchTable);

  return 0;
}

void __stdcall ServiceMain(DWORD dwArgc, char *lpszArgv)
{
  g_ServiceStatusHandle = ::RegisterServiceCtrlHandler(g_szServiceName, ServiceCtrlHandle);

  TellSCM(SERVICE_START_PENDING, 0, 1);
  TellSCM(SERVICE_RUNNING, 0, 0);

  while (TRUE)
  {
    Sleep(5000);
    DoTask();
  }
}

void __stdcall ServiceCtrlHandle(DWORD dwOperateCode)
{
  switch (dwOperateCode)
  {
  case SERVICE_CONTROL_PAUSE:
  {
    // 暫停
    TellSCM(SERVICE_PAUSE_PENDING, 0, 1);
    TellSCM(SERVICE_PAUSED, 0, 0);
    break;
  }
  case SERVICE_CONTROL_CONTINUE:
  {
    // 繼續
    TellSCM(SERVICE_CONTINUE_PENDING, 0, 1);
    TellSCM(SERVICE_RUNNING, 0, 0);
    break;
  }
  case SERVICE_CONTROL_STOP:
  {
    // 停止
    TellSCM(SERVICE_STOP_PENDING, 0, 1);
    TellSCM(SERVICE_STOPPED, 0, 0);
    break;
  }
  case SERVICE_CONTROL_INTERROGATE:
  {
    // 詢問
    break;
  }
  default:
    break;
  }
}

BOOL TellSCM(DWORD dwState, DWORD dwExitCode, DWORD dwProgress)
{
  SERVICE_STATUS serviceStatus = { 0 };
  BOOL bRet = FALSE;

  ::RtlZeroMemory(&serviceStatus, sizeof(serviceStatus));
  serviceStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
  serviceStatus.dwCurrentState = dwState;
  serviceStatus.dwControlsAccepted = SERVICE_ACCEPT_STOP | SERVICE_ACCEPT_PAUSE_CONTINUE | SERVICE_ACCEPT_SHUTDOWN;
  serviceStatus.dwWin32ExitCode = dwExitCode;
  serviceStatus.dwWaitHint = 3000;

  bRet = ::SetServiceStatus(g_ServiceStatusHandle, &serviceStatus);
  return bRet;
}

void DoTask()
{
  // 自己程式實現部分代碼放在這裡
}

設置SCM開機運行

獨立的SCM程式無法直接雙擊運行,該服務程式只能通過服務管理器運行,通過使用CreateService將服務管理器程式設置為開機自動運行,並使用StartService將服務啟動。

CreateService 函數用於創建一個新的服務。這個函數通常在安裝服務時使用。在服務安裝過程中,需要指定服務的名稱、顯示名稱、服務類型、啟動類型、二進位路徑等信息。

以下是 CreateService 函數的原型:

SC_HANDLE CreateService(
  SC_HANDLE hSCManager,
  LPCTSTR   lpServiceName,
  LPCTSTR   lpDisplayName,
  DWORD     dwDesiredAccess,
  DWORD     dwServiceType,
  DWORD     dwStartType,
  DWORD     dwErrorControl,
  LPCTSTR   lpBinaryPathName,
  LPCTSTR   lpLoadOrderGroup,
  LPDWORD   lpdwTagId,
  LPCTSTR   lpDependencies,
  LPCTSTR   lpServiceStartName,
  LPCTSTR   lpPassword
);
  • hSCManager: 服務控制管理器的句柄,可以通過 OpenSCManager 函數獲取。
  • lpServiceName: 要創建的服務的名稱。這是服務在 SCM 中的唯一標識符。
  • lpDisplayName: 服務的顯示名稱,這是在服務列表中顯示的名稱。
  • dwDesiredAccess: 對服務的訪問許可權,例如 SERVICE_ALL_ACCESS
  • dwServiceType: 服務的類型,例如 SERVICE_WIN32_OWN_PROCESS
  • dwStartType: 服務的啟動類型,例如 SERVICE_AUTO_START
  • dwErrorControl: 當服務無法啟動時的錯誤處理控制。
  • lpBinaryPathName: 服務程式的可執行文件的路徑。
  • lpLoadOrderGroup: 指定服務應屬於的載入順序組。
  • lpdwTagId: 指向接收服務標識符的指針。
  • lpDependencies: 指定服務依賴的服務名稱。
  • lpServiceStartName: 服務啟動時使用的用戶名。
  • lpPassword: 服務啟動時使用的密碼。

函數返回一個 SC_HANDLE 類型的句柄,該句柄標識了新創建的服務。如果函數調用失敗,返回 NULL。可以通過調用 GetLastError 函數獲取錯誤代碼。

StartService 函數用於啟動一個已註冊的服務。這個函數通常在服務程式中的啟動代碼或者通過服務管理工具中手動啟動服務時使用。

以下是 StartService 函數的原型:

BOOL StartService(
  SC_HANDLE hService,
  DWORD     dwNumServiceArgs,
  LPCTSTR   *lpServiceArgVectors
);
  • hService: 要啟動的服務的句柄,可以通過 OpenService 函數獲取。
  • dwNumServiceArgs: 指定傳遞給服務的命令行參數數量。
  • lpServiceArgVectors: 指向包含服務命令行參數的字元串數組。

函數返回一個 BOOL 類型的值,如果調用成功返回 TRUE,否則返回 FALSE。可以通過調用 GetLastError 函數獲取錯誤代碼。

ControlService 函數用於向已註冊的服務發送控制碼,以便執行特定的操作。這個函數通常在服務程式中的控制邏輯或者通過服務管理工具中手動控制服務時使用。

以下是 ControlService 函數的原型:

BOOL ControlService(
  SC_HANDLE        hService,
  DWORD            dwControl,
  LPSERVICE_STATUS lpServiceStatus
);
  • hService: 要控制的服務的句柄,可以通過 OpenService 函數獲取。
  • dwControl: 指定服務的控制碼,可以是以下之一:
    • SERVICE_CONTROL_CONTINUE: 繼續服務。
    • SERVICE_CONTROL_PAUSE: 暫停服務。
    • SERVICE_CONTROL_STOP: 停止服務。
    • 等等,還有其他服務控制碼。
  • lpServiceStatus: 指向 SERVICE_STATUS 結構體的指針,用於接收服務的當前狀態信息。

函數返回一個 BOOL 類型的值,如果調用成功返回 TRUE,否則返回 FALSE。可以通過調用 GetLastError 函數獲取錯誤代碼。

如下代碼實現了服務管理的兩個關鍵功能:AutoRunService 函數用於註冊並啟動服務,使其在系統啟動時自動運行;SetServiceStatus 函數用於設置服務的狀態,包括停止服務、啟動服務和刪除服務。

這樣的功能對於管理系統服務的狀態和自啟動行為具有重要意義。然而,需要註意確保在執行這些操作時具有足夠的許可權,併在實際應用中加強錯誤處理以確保操作的可靠性。

#include <stdio.h>
#include <Windows.h>
#include <Shlwapi.h>

#pragma comment(lib, "Shlwapi.lib")

// 註冊服務自啟動
void AutoRunService(char* szFilePath, char* szDescribe)
{
  char szName[MAX_PATH] = { 0 };
  SC_HANDLE scHandle = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
  lstrcpy(szName, szFilePath);
  PathStripPath(szName);

  SC_HANDLE scHandleOpen = OpenService(scHandle, szName, SERVICE_ALL_ACCESS);
  if (scHandleOpen == NULL)
  {
    // SERVICE_AUTO_START => 隨系統自動啟動 | SERVICE_DEMAND_START => 手動啟動
    SC_HANDLE scNewHandle = CreateService(scHandle, szName, szDescribe,
      SERVICE_ALL_ACCESS, SERVICE_WIN32_OWN_PROCESS, SERVICE_AUTO_START,
      SERVICE_ERROR_IGNORE, szFilePath, NULL, NULL, NULL, NULL, NULL);
    StartService(scNewHandle, 0, NULL);
    CloseServiceHandle(scNewHandle);
    printf("[*] 創建服務完成 \n");
  }
  CloseServiceHandle(scHandleOpen);
  CloseServiceHandle(scHandle);
}

// 設置服務狀態
BOOL SetServiceStatus(char* szName, int Status)
{
  SERVICE_STATUS ss;
  SC_HANDLE scHandle = OpenSCManager(NULL, NULL, SC_MANAGER_ALL_ACCESS);
  SC_HANDLE scHandleOpen = OpenService(scHandle, szName, SERVICE_ALL_ACCESS);
  BOOL bRet = TRUE;
  if (scHandleOpen != NULL)
  {
    switch (Status)
    {
    case 1: if (!ControlService(scHandleOpen, SERVICE_CONTROL_STOP, &ss)) { bRet = FALSE; }; break;
    case 2: if (!StartService(scHandleOpen, 0, NULL)) { bRet = FALSE; }; break;
    case 3: if (!DeleteService(scHandleOpen)) { bRet = FALSE; }break;
    default:break;
    }
  }
  CloseServiceHandle(scHandleOpen);
  CloseServiceHandle(scHandle);
  return bRet;
}

int main(int argc, char* argv[])
{
  // 註冊為自啟動服務將d:/myservice.exe 註冊為自啟動服務 後面是描述信息
  AutoRunService((char *)"d:/myservice.exe", (char *)"Microsoft Windows Security Services");

  // 根據服務名稱管理服務 1=>停止服務 2=>啟動服務 3=>刪除服務
  BOOL ret = SetServiceStatus((char *)"myservice.exe", 2);
  printf("狀態: %d \n", ret);

  system("pause");
  return 0;
}

運行上述代碼將自動把d:/myservice.exe添加至服務自啟動列表,並可以通過枚舉的方式找到該服務的具體信息,如下圖所示;

文章出處:https://www.cnblogs.com/LyShark/p/17850974.html
本博客所有文章除特別聲明外,均採用 BY-NC-SA 許可協議。轉載請註明出處!
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Hi i,m JinXiang ⭐ 前言 ⭐ 本篇文章主要介紹Java時間的幾種常見方法以及部分理論知識 🍉歡迎點贊 👍 收藏 ⭐留言評論 📝私信必回喲😁 🍉博主收將持續更新學習記錄獲,友友們有任何問題可以在評論區留言 ⭐什麼是Date日期類型? 在JDK1.0中,Date類是唯一的一個代 ...
  • 1.pom文件增加依賴 <dependency> <groupId>com.github.ulisesbocchio</groupId> <artifactId>jasypt-spring-boot-starter</artifactId> <version>3.0.5</version> </de ...
  • 在數據處理和報告生成的領域中,Excel 文件一直是廣泛使用的標準格式。為了讓 Python 開發者能夠輕鬆創建和修改 Excel 文件,XlsxWriter 庫應運而生。XlsxWriter 是一個功能強大的 Python 模塊,專門用於生成 Microsoft Excel 2007及以上版本(.... ...
  • 函數式(Functional)介面 只包含一個抽象方法的介面,稱為函數式介面。 你可以通過Lambda表達式來創建該介面的對象。(若Lambda表達式拋出一個受檢異常(即:非運行時異常),那麼該異常需要在目標介面的抽象方法上進行聲明 我們可以在一個介面上使用@Functionallnterface註 ...
  • 問題出現的前提 keycloak通過k8s部署,併進行了集群部署,共2個節點 通過功能變數名稱解析後,直接到外網LB,在LB上配置了k8s-ingress的IP,埠是80和443 在keycloak應用的ingress配置中,對功能變數名稱進行了keycloak服務的綁定 問題的描述 有時間無法完成登錄,點登錄後 ...
  • 介紹了Spring的高級註解,包括@Configuration註解替代XML配置文件、@Bean註解創建和註入Bean、@ComponentScan註解進行註解掃描,配置Bean的底層實現原理等 ...
  • java三種類型的加減,LocalDate、Calendar、Date @目錄1.LocalDate類型加減:2.Calendar加減:3.Date類型加減 1.LocalDate類型加減: 以下是LocalDate類進行日期加減: import java.time.LocalDate; impor ...
  • Date類: 1.1、將字元串型時間日期轉化為date類型 String timeString = "2023-11-17 09:27:00"; SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //創建"簡單時間 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...