程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> VC >> 關於VC++ >> 如何用Win32 APIs枚舉應用程序窗口和進程

如何用Win32 APIs枚舉應用程序窗口和進程

編輯:關於VC++

摘要

我們在編寫程序時,常常遇到的一件事情就是要准確列出系統中所有正在運行的程序或者進程。Windows 任務管理器就是這樣的一個程序。它既能列出運行的桌面應用程序,又能列出系統中所有運行的進程。那麼,我們在程序中如何實現這樣的任務呢?本文下面將詳細討論這個問題。

枚舉頂層(top-level)窗口

枚舉桌面頂層窗口相對於枚舉進程來說可能要容易一些。枚舉桌面頂層窗口的方法是用 EnumWindows() 函數。不要用 GetWindow()來創建窗口列表,因為窗口之間復雜的父子及同胞關系(Z-Order)容易造成混亂而使得枚舉結果不准確。

EnumWindows()有兩個參數,一個是指向回調函數的指針,一個是用戶定義的 LPARAM 值, 針對每個桌面窗口(或者頂層窗口)它調用回調函數一次。然後回調函數用該窗口句柄做一些處理,比如將它添加到列表中。這個方法保證枚舉結果不會被窗口復雜的層次關系搞亂,因此,一旦有了窗口句柄,我們就可以通過 GetWindowText() 得到窗口標題。

枚舉進程

建立系統進程列表比枚舉窗口稍微復雜一些。這主要是因為所用的 API 函數對於不同的 Win32 操作系統有依賴性。在 Windows 9x、Windows Me、Windows 2000 Professional 以及 Windows XP 中,我們可以用 ToolHelp32 庫中的 APIs 函數。但是在 Windows NT 裡,我們必須用 PSAPI 庫中的 APIs 函數, PSAPI 庫是 SDK 的一部分。本文我們將討論上述所有平台中的實現。附帶的例子程序將對上述庫中的 APIs 進行包裝,以便包裝後的函數能支持所有 Win32 操作系統。

使用 ToolHelp32 庫枚舉進程

ToolHelp32 庫函數在 KERNEL32.dll 中,它們都是標准的 API 函數。但是 Windows NT 4.0 不提供這些函。

ToolHelp32 庫中有各種各樣的函數可以用來枚舉系統中的進程、線程以及獲取內存和模塊信息。其中枚舉進程 只需用如下三個的函數:CreateToolhelp32Snapshot()、Process32First()和 Process32Next()。

使用 ToolHelp32 函數的第一步是用 CreateToolhelp32Snapshot() 函數創建系統信息“快照”。這個函數可以讓你選擇存儲在快照中的信息類型。如果你只是對進程信息感興趣,那麼只要包含 TH32CS_SNAPPROCESS 標志即可。 CreateToolhelp32Snapshot() 函數返回一個 HANDLE,完成調用之後,必須將此 HANDLE 傳給 CloseHandle()。

接下來是調用一次 Process32First 函數,從快照中獲取進程列表,然後重復調用 Process32Next,直到函數返回 FALSE 為止。這樣將遍歷快照中進程列表。這兩個函數都帶兩個參數,它們分別是快照句柄和一個  PROCESSENTRY32 結構。

調用完 Process32First 或 Process32Next 之後,PROCESSENTRY32 中將包含系統中某個進程的關鍵信息。其中進程 ID 就存儲在此結構的 th32ProcessID。此 ID 可以被傳給 OpenProcess() API 以獲得該進程的句柄。對應的可執行文件名及其存放路徑存放在 szExeFile 結構成員中。在該結構中還可以找到其它一些有用的信息。

注意:在調用 Process32First() 之前,一定要記住將 PROCESSENTRY32 結構的 dwSize 成員設置成 sizeof(PROCESSENTRY32)。

使用 PSAPI 庫枚舉進程

在 Windows NT 中,創建進程列表使用 PSAPI 函數,這些函數在 PSAPI.DLL 中。這個文件是隨 Platform SDK 一起分發的,最新版本的 Platform SDK 可以從這裡下載:

使用這個庫所需的 PSAPI.h 和 PSAPI.lib 文件也在該 Platform SDK 中。

為了使用 PSAPI 庫中的函數,需將 PSAPI.lib 添加到代碼項目中,同時在所有調用 PSAPI API 的模塊中包含 PSAPI.h 文件。記住一定要隨可執行文件一起分發 PSAPI.DLL,因為它不隨 Windows NT 一起分發。你可以點擊這裡單獨下載 PSAPI.DLL 的可分發版本(不用完全下載 Platform SDK)。

與 ToolHelp32 一樣,PSAPI 庫也包含各種各樣有用的函數。由於篇幅所限,本文只討論與枚舉進程有關函數:EnumProcesses()、EnumProcessModules()、GetModuleFileNameEx()和 GetModuleBaseName()。

創建進程列表的第一步是調用 EnumProcesses()。該函數的聲明如下:

BOOL EnumProcesses( DWORD *lpidProcess, DWORD cb, DWORD *cbNeeded );

EnumProcesses()帶三個參數,DWORD 類型的數組指針 lpidProcess;該數組的大小尺寸 cb;以及一個指向 DWORD 的指針 cbNeeded,它接收返回數據的長度。DWORD 數組用於保存當前運行的進程 IDs。cbNeeded 返回數組所用的內存大小。下面算式可以得出返回了多少進程:nReturned = cbNeeded / sizeof(DWORD)。

注意:雖然文檔將返回的 DWORD 命名為“cbNeeded”,實際上是沒有辦法知道到底要傳多大的數組的。EnumProcesses()根本不會在 cbNeeded 中返回一個大於 cb 參數傳遞的數組值。結果,唯一確保 EnumProcesses()函數成功的方法是分配一個 DWORD 數組,並且,如果返回的 cbNeeded 等於 cb,分配一個較大的數組,並不停地嘗試直到 cbNeeded 小於 cb

現在,你獲得了一個數組,其元素保存著系統中每個進程的ID。如果你要想獲取進程名,那麼你必須首先獲取一個句柄。要想從進程 ID 得到句柄,就得調用 OpenProcess()。

一旦有了句柄,則需要得到該進程的第一個模塊。為此調用 EnumProcessModules() API:EnumProcessModules( hProcess, &hModule, sizeof(hModule), &cbReturned );  調用之後,hModule 變量中保存的將是進程中的第一個模塊。記住進程其實沒有名字,但進程的第一個模塊既是該進程的可執行模塊。現在你可以用 hModule 中返回的模塊句柄調用 GetModuleFileNameEx() 或 GetModuleBaseName() API 函數獲取全路徑名,或者僅僅是進程可執行模塊名。兩個函數均帶四個參數:進程句柄,模塊句柄,返回名字的緩沖指針以及緩沖大小尺寸。

用 EnumProcesses() API 返回的每一個進程 ID 重復這個調用過程,你便可以創建 Windows NT 的進程列表。

16位進程的處理方法

在 Windows 95,Windows 98 和 Windows ME 中,ToolHelp32 對待16位程序一視同仁,它們與 Win32 程序一樣有自己的進程IDs。但是在 Windows NT,Windows 2000 或 Windows XP 中情況並不是這樣。在這些操作系統中,16位程序運行在所謂的 VDM 當中(也就是DOS機)。

為了在 Windows NT,Windows 2000 和 Windows XP 中枚舉16位程序,你必須使用一個名為 VDMEnumTaskWOWEx()的函數。在源代碼模塊中必須包含 VDMDBG.h,並且 VDMDBG.lib 文件必須與項目鏈接。這兩個文件都在 Platform SDK 中。該函數的聲明如下:INT WINAPI VDMEnumTaskWOWEx( DWORD dwProcessId, TASKENUMPROCEX fp,LPARAM lparam );

此處 dwProcessId 是 NTVDM 中擬枚舉的16位任務進程標示符。參數 fp 是回調枚舉函數的指針。參數 lparam 是用戶定義的值,它被傳遞到枚舉函數。枚舉函數應該被定義成如下這樣:

BOOL WINAPI Enum16( DWORD dwThreadId,
          WORD hMod16,
          WORD hTask16,
          PSZ pszModName,
          PSZ pszFileName,
          LPARAM lpUserDefined );
  該函數針對每個運行在 NTVDM 進程中的16位任務調用一次,NTVDM 進程ID將被傳入 VDMEnumTaskWOWEx()。如果想繼續枚舉則返回 FALSE,終止枚舉則返回 TRUE。注意這是與 EnumWindows()相對的。

關於代碼

本文附帶的代碼例子將 PSAPI 和 ToolHelp32 封裝到一個名為 EnumProcs() 的函數中。該函數的工作原理類似 EnumWindows(),有一個指向回調函數的指針,並要對該函數進行重復調用,針對系統中的每個進程調用一次。另一個參數是用戶定義的 lParam。下面是該函數的聲明:BOOL WINAPI EnumProcs( PROCENUMPROC lpProc, LPARAM lParam );

使用該函數時,要象下面這樣聲明回調函數:

BOOL CALLBACK Proc( DWORD dw, WORD w16, LPCSTR lpstr, LPARAM lParam );

參數 dw 包含 ID,“w16”是16位任務的任務號,如果為32位進程則為0(在 Windows 95 中總是0),參數lpstr 指向文件名,lParam 是用戶定義的,要被傳入 EnumProcs()。

EnumProcs() 函數通過顯示鏈接使用 ToolHelp32 和 PSAPI,而非通常所用的隱式鏈接。之所以要這樣做,主要是為了讓代碼能夠在二進制一級兼容,從可以在所有 Win32 操作系統平台上運行。

本文配套源碼

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved