getmodulehandleex函數詳解

一、函數概述

GetModuleHandleEx函數是Windows操作系統提供的一個函數,用於獲取指定模塊的句柄。該函數被廣泛應用於Windows編程中,是操作系統提供的一個非常強大的函數。

二、函數原型和參數

BOOL GetModuleHandleEx(
  DWORD   dwFlags,
  LPCTSTR lpModuleName,
  HMODULE *phModule
);

GetModuleHandleEx函數的參數如下:

  • dwFlags:一個DWORD類型的標誌,標誌着以何種方式獲取模塊句柄。這裡可選的常量標誌有:GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS,GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,GET_MODULE_HANDLE_EX_FLAG_PIN。
  • lpModuleName:一個指向字符串的指針,用於指定模塊的名字。
  • phModule:一個HMODULE類型的指針,函數返回時用於接收穫取到的模塊句柄。

三、獲取當前進程的模塊句柄

使用GetModuleHandleEx函數可以獲取當前進程中指定模塊的句柄。例如,下面的代碼用於獲取當前進程中kernel32.dll模塊的句柄:

DWORD dwRet = 0;
HMODULE hKernel32 = NULL;

// 獲取kernel32模塊句柄
dwRet = GetModuleHandleEx(
    GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
    (LPCTSTR)&GetModuleHandleEx,
    &hKernel32);

通過這段代碼,我們可以獲得當前進程中kernel32模塊的句柄。

四、獲取其他進程的模塊句柄

GetModuleHandleEx函數同樣可以用於獲取其他進程中指定模塊的句柄。為了安全起見,我們需要先打開指定進程的句柄,然後在該進程中調用GetModuleHandleEx函數來獲取指定模塊的句柄。下面的代碼演示了如何獲取其他進程中kernel32.dll模塊的句柄:

// 首先打開指定進程的句柄
HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwProcessId);
if (NULL == hProcess)
{
    return FALSE;
}

// 在指定進程中獲取kernel32模塊句柄
dwRet = GetModuleHandleEx(
    GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
    (LPCTSTR)&GetModuleHandleEx,
    &hKernel32);

// 關閉指定進程的句柄
CloseHandle(hProcess);

通過這段代碼,我們可以獲得指定進程中kernel32模塊的句柄。

五、給其他進程增加引用計數

在獲取其他進程中指定模塊的句柄時,如果我們需要在其他進程中進行操作,首先需要給該句柄增加引用計數。例如,下面的代碼演示了如何獲取其他進程中的kernel32.dll模塊,然後調用其中的函數:

// 首先打開指定進程的句柄
HANDLE hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, dwProcessId);
if (NULL == hProcess)
{
    return FALSE;
}

// 在指定進程中獲取kernel32模塊句柄
dwRet = GetModuleHandleEx(
    GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT,
    (LPCTSTR)&GetModuleHandleEx,
    &hKernel32);

// 增加引用計數
Module32First(hSnapshot, &me32);
while (Module32Next(hSnapshot, &me32))
{
    if (me32.hModule == hKernel32)
    {
        m_info.th32ModuleID = me32.th32ModuleID; // 模塊ID
        m_info.hModule = me32.hModule;           // 模塊句柄
        m_info.modBaseAddr = me32.modBaseAddr;   // 模塊基址
        m_info.modBaseSize = me32.modBaseSize;   // 模塊大小
        HANDLE hModule = OpenProcess(PROCESS_ALL_ACCESS, FALSE, me32.th32ProcessID);
        if (hModule)
        {
            DWORD dwResult = 0;
            // 如果引用計數增加成功,則返回該模塊的句柄,否則返回NULL。
            if (TRUE == Module32First(hSnapshot, &me32))
            {
                dwRet = Module32First(hModule, &me32);
                if (dwRet == TRUE)
                {
                    while (TRUE == Module32Next(hModule, &me32))
                    {
                        if (me32.hModule == hKernel32)
                        {
                            HANDLE hProc = OpenProcess(PROCESS_ALL_ACCESS, FALSE, me32.th32ProcessID);
                            LoadLibraryA(m_info.szExePath);
                            CloseHandle(hProc);
                            break;
                        }
                    }
                }
            }
            CloseHandle(hModule);
        }
        break;
    }
}

// 關閉指定進程的句柄
CloseHandle(hProcess);

在調用GetModuleHandleEx函數獲取模塊句柄之後,我們通過Module32First和Module32Next函數遍歷進程模塊,找到指定模塊。然後我們調用OpenProcess打開該模塊所在進程的句柄,並調用LoadLibraryA函數增加該模塊的引用計數。增加成功之後,我們就可以在其他進程中安全地訪問該模塊了。

總結

GetModuleHandleEx函數是Windows操作系統提供的一個非常實用的函數,可以用於獲取指定模塊的句柄,並且還支持在其他進程中獲取句柄並增加引用計數。通過本文的介紹,我們相信大家已經了解了GetModuleHandleEx函數的基本用法,並能夠在實際Windows編程中使用它來解決問題。

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/242739.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-12 12:51
下一篇 2024-12-12 12:51

相關推薦

  • Python中引入上一級目錄中函數

    Python中經常需要調用其他文件夾中的模塊或函數,其中一個常見的操作是引入上一級目錄中的函數。在此,我們將從多個角度詳細解釋如何在Python中引入上一級目錄的函數。 一、加入環…

    編程 2025-04-29
  • Python中capitalize函數的使用

    在Python的字符串操作中,capitalize函數常常被用到,這個函數可以使字符串中的第一個單詞首字母大寫,其餘字母小寫。在本文中,我們將從以下幾個方面對capitalize函…

    編程 2025-04-29
  • Python中set函數的作用

    Python中set函數是一個有用的數據類型,可以被用於許多編程場景中。在這篇文章中,我們將學習Python中set函數的多個方面,從而深入了解這個函數在Python中的用途。 一…

    編程 2025-04-29
  • 三角函數用英語怎麼說

    三角函數,即三角比函數,是指在一個銳角三角形中某一角的對邊、鄰邊之比。在數學中,三角函數包括正弦、餘弦、正切等,它們在數學、物理、工程和計算機等領域都得到了廣泛的應用。 一、正弦函…

    編程 2025-04-29
  • 單片機打印函數

    單片機打印是指通過串口或並口將一些數據打印到終端設備上。在單片機應用中,打印非常重要。正確的打印數據可以讓我們知道單片機運行的狀態,方便我們進行調試;錯誤的打印數據可以幫助我們快速…

    編程 2025-04-29
  • Python3定義函數參數類型

    Python是一門動態類型語言,不需要在定義變量時顯示的指定變量類型,但是Python3中提供了函數參數類型的聲明功能,在函數定義時明確定義參數類型。在函數的形參後面加上冒號(:)…

    編程 2025-04-29
  • Python定義函數判斷奇偶數

    本文將從多個方面詳細闡述Python定義函數判斷奇偶數的方法,並提供完整的代碼示例。 一、初步了解Python函數 在介紹Python如何定義函數判斷奇偶數之前,我們先來了解一下P…

    編程 2025-04-29
  • Python實現計算階乘的函數

    本文將介紹如何使用Python定義函數fact(n),計算n的階乘。 一、什麼是階乘 階乘指從1乘到指定數之間所有整數的乘積。如:5! = 5 * 4 * 3 * 2 * 1 = …

    編程 2025-04-29
  • Python函數名稱相同參數不同:多態

    Python是一門面向對象的編程語言,它強烈支持多態性 一、什麼是多態多態是面向對象三大特性中的一種,它指的是:相同的函數名稱可以有不同的實現方式。也就是說,不同的對象調用同名方法…

    編程 2025-04-29
  • 分段函數Python

    本文將從以下幾個方面詳細闡述Python中的分段函數,包括函數基本定義、調用示例、圖像繪製、函數優化和應用實例。 一、函數基本定義 分段函數又稱為條件函數,指一條直線段或曲線段,由…

    編程 2025-04-29

發表回復

登錄後才能評論