如何在我的命名空间中使用 LoadString 宏?

How can I use the LoadString macros in my namespace?

我在命名空间中使用 LoadString WinAPI 宏时遇到问题。我的函数:

namespace Bushman {
    // Get the string from the resource's string table of the module.
    // I use the same name like WinAPI macros but with own signature.
    PTSTR LoadString(HMODULE h, DWORD id) {
        h = NULL == h ? ::GetModuleHandle(NULL) : h;
        PTSTR ptr = NULL;
        // it returns really length instead of concatenated 
        // string length, therefore I can use it for malloc.
        int i = ::LoadString(h, id, (PTSTR)&ptr, 0);
        if (0 == i) {
            return NULL;
        }
        PTSTR string = (PTSTR)malloc(i * (sizeof(TCHAR) + 1));
        ::LoadString(h, id, string, i + 1);
        return string; // NOTE: don't forget free resource in the outer code.
    }
}

我得到编译错误:

'LoadStringW': is not a member of 'Bushman'
'LoadStringW': is not a member of 'Bushman'
'LoadStringW': is not a member of 'Bushman'

我该如何解决?

UPD

我认为问题是宏有这样的定义

#ifdef UNICODE
#define LoadString  LoadStringW
#else
#define LoadString  LoadStringA
#endif // !UNICODE

而不是像这样:

#ifdef UNICODE
#define LoadString  ::LoadStringW
#else
#define LoadString  ::LoadStringA
#endif // !UNICODE

UPD 2

我找到问题的原因了。问题出在我的代码的其他地方。我在我的代码中使用了这样的声明:

namespace Bushman {} // namespace declaration
PTSTR Bushman::LoadString(HMODULE h, DWORD id); // function declaration

但这是错误的。如果我将其重写为:

,一切正常
namespace Bushman {
    PTSTR LoadString(HMODULE h, DWORD id);
}

您有几个选择:

  1. 不要包含定义 LoadString 宏的 Windows 头文件,或者
  2. 包含该头文件,但使用 #undef 取消定义宏。

这个问题真的没有很好的解决办法。一旦你开始使用宏,你就失去了轻松隔离和控制它们影响的能力。预处理器不关心你的命名空间。


关于你更新的问题,你并没有真正解决问题。从 UPD2 中的代码开始,预处理器将 LoadString 转换为 LoadStringW。您只是没有意识到这一点,因为它在编译过程中透明地发生。但是如果你尝试从另一个没有定义 LoadString 宏的翻译单元使用你的 class 你会发现这个函数被命名为 LoadStringW.