在非mfc应用程序中强制从dll加载资源的WTL方式? (我们使用的是WTL / ATL,而不是直接的win32)

时间:2011-03-17 16:11:07

标签: c++ winapi resources atl wtl

我之前发布了this question,现在加载了本地化的字符串(我们用LoadString()获得的那些)但我还需要从附属DLL加载所有其他资源。

MFC有AfxSetResourceHandle()调用,但我需要一些非mfc应用程序的东西吗?我怀疑我必须在初始化代码中设置它,所以我的所有资源都是从另一个DLL加载的。我如何在WTL(Windows模板库)上下文中执行此操作?

编辑:

This summarizes our problem

我们不是直接使用win32,而是ATL和WTL用于Windows的东西。所以我们不能依赖MFC的东西,我们没有低级别的菜单和对话框资源加载控制。

编辑: 嗯... This seems to have an answer,但我希望有更好的东西。例如 - 一个SetResourceInstance()方法,类似于CAppModule对象中的GetResourceInstance()。

2 个答案:

答案 0 :(得分:6)

资源函数(FindResource,LoadResource)将模块的句柄作为参数之一。

使用GetModuleHandleEx获取DLL的模块句柄。

编辑:ATL / WTL的其他信息。

WTL在其Win32调用中使用ATL::_AtlBaseModule.GetResourceInstance()作为模块句柄。有一个SetResourceInstance函数,您可以调用它来更改使用的模块。这样的事情应该在你的程序开始时起作用:

HMODULE hmod;
::GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_PIN, myDllFuncPtr, &hmod);
ATL::_AtlBaseModule.SetResourceInstance(hmod);

答案 1 :(得分:0)

有时上述方法无法使用,例如当您因某些原因仍需支持Windows 2000时。在这种情况下,有以下技巧很方便。

我们声明一个静态变量,这意味着它的地址将在它所链接的模块内。然后我们使用该变量的地址来查询该分配区域的基地址,这就是HMODULE的内容。

HMODULE GetCurrentModuleHandle()
{
    MEMORY_BASIC_INFORMATION mbi;
    static int iDummy;
    VirtualQuery(&iDummy, &mbi, sizeof(mbi));
    return (HMODULE)mbi.AllocationBase;
}

这绝不会使Mark的答案无效!如果你需要你的程序在古老的系统上运行,请记住它作为后备选项。