使用 C++ 在当前进程中获取指定模块的基址

C++ 实现 , 获取指定模块在该进程中的基址

1、流程:

获取进程的所有模块信息-->遍历模块列表

2、实现

cpp 复制代码
// 我自己定义的
typedef struct moudle_date_ {
	HANDLE mhandle;   // 句柄
	char mname[64];   // 名称
	char* date;       // 数据
	DWORD mdword;     // 基址 
}moudle_date;

// 传入该进程的 pid 、需要查找模块的名称
moudle_date* getMoudle(DWORD pid, const char* name) {
	moudle_date* date = nullptr;
	/*
    CreateToolhelp32Snapshot()->获取指定进程以及这些进程使用的堆、模块和线程的快照。
    TH32CS_SNAPALL -> 包括系统中的所有进程和线程,以及 pid 中指定的进程的堆和模块。
    return value : 如果函数成功,它将返回指定快照的打开句柄。如果函数失败,它将返回 INVALID_HANDLE_VALUE
    */
	HANDLE handle = CreateToolhelp32Snapshot( TH32CS_SNAPALL,  pid );

	std::cout << "getMoudle::handle :" << std::hex<<handle << std::endl;
	if ( handle == INVALID_HANDLE_VALUE) {
		std::cout << "Error CreateToolhelp32Snapshot" << std::endl;
		return nullptr;
	}
	/*
	MODULEENTRY32:Windows API 中定义的一个结构体,用于存储模块的信息。
	*/
	MODULEENTRY32 info;
	memset(&info, 0, sizeof(info));
    // dwSize 结构体的大小,一定要设置,为 Module32First、Module32Next做准备
	info.dwSize = sizeof(MODULEENTRY32);

	char sname[256];
	strcpy_s(sname, sizeof(sname) , name);

	/*
	循环遍历模块 ,找到基址
	Module32First() : 检索与进程关联的第一个模块的相关信息。
	传入参数: CreateToolhelp32Snapshot返回的进程句柄、info(MODULEENTRY32)
	返回值 : bool类型,成功返回true. 失败返回false.
	*/
	bool ret = Module32First( handle, &info);
	if (!ret) {
        /*
        GetLastError(): 获取错误代码.
        返回类型 DWORD.
        */
		DWORD err = GetLastError();
		std::cout << "getMoudle::GetLastError : "<<err << std::endl;
		return nullptr;
	}
	while ( ret ) {
		// szModule -> 模块的名称
        // 注意: szModule的类型是 wchar_t 宽字节
        // 我们传入的模块名称 类型是 char ,所以需要进行类型转换,才能够比较
		if ( strcmp( UnicodeToUtf8(info.szModule) , sname)== 0 ) {
			date = new moudle_date;
            // modBaseAddr ->MODULEENTRY32结构体中的成员,模块的基址
			date->mdword = (DWORD)info.modBaseAddr;
			CloseHandle(handle);  // 清理资源
			return date;
		}
		/*
		Module32Next(): 获取下一个模块的信息
		*/
		ret = Module32Next( handle, &info);
	}
	return nullptr;
}

补充:UnicodeToUtf8 -> 如何将宽字符字符串转换为 UTF-8 编码的字符串UnicodeToUtf8

这些你可以不需要知道怎么实现的,直接拿来用就好了,没有人会去记这些东西。

cpp 复制代码
char* UnicodeToUtf8(const wchar_t* lpszStr)
{
	char* lpUtf8;
	int nLen;

	if (NULL == lpszStr)
		return NULL;

	nLen = ::WideCharToMultiByte(CP_UTF8, 0, lpszStr, -1, NULL, 0, NULL, NULL);
	if (0 == nLen)
		return NULL;

	lpUtf8 = new char[nLen + 1];
	if (NULL == lpUtf8)
		return NULL;
	memset(lpUtf8, 0, nLen + 1);
	nLen = ::WideCharToMultiByte(CP_UTF8, 0, lpszStr, -1, lpUtf8, nLen, NULL, NULL);
	if (0 == nLen)
	{
		delete[]lpUtf8;
		return NULL;
	}
	return lpUtf8;
}

希望对你有所帮助

相关推荐
We་ct3 分钟前
深度剖析浏览器跨域问题
开发语言·前端·浏览器·跨域·cors·同源·浏览器跨域
skywalk816312 分钟前
在考虑双轨制,即在中文语法的基础上,加上数学公式的支持,这样像很多计算将更加简单方便,就像现在的小学数学课本里面一样,比如:定x=2*x + 1
开发语言
小书房15 分钟前
Kotlin的by
android·开发语言·kotlin·委托·by
王老师青少年编程27 分钟前
csp信奥赛C++高频考点专项训练之贪心算法 --【贪心与二分判定】:数列分段 Section II
c++·算法·贪心·csp·信奥赛·二分判定·数列分段 section ii
zh_xuan29 分钟前
libcurl调用https接口
c++·libcurl
就叫飞六吧31 分钟前
QT写一个桌面程序exe并动态打包基本流程(c++)
开发语言·c++
蜡笔小马32 分钟前
1.c++设计模式-工厂模式
c++
threelab40 分钟前
Three.js 代码云效果 | 三维可视化 / AI 提示词
开发语言·javascript·人工智能
V搜xhliang02461 小时前
OpenClaw科研全场景用法:从文献到实验室的完整自动化方案
运维·开发语言·人工智能·python·算法·microsoft·自动化
kaikaile19951 小时前
风、浪、流环境模型的船舶三自由度(纵荡、横荡、艏摇)运动仿真MATLAB
开发语言·人工智能·matlab