mingw:从 lambda 到函数指针的转换无效

mingw: Invalid conversion from lambda to function pointer

本文关键字:指针 转换 无效 函数 lambda mingw      更新时间:2023-10-16
使用

mingw 5.3.0 进行编译时,尝试提供 lambda 作为指向 Win32 API 调用的函数指针失败,尽管在使用 MSVC(Visual Studio 2013)时编译正常。

MCVE:

#include <Windows.h>
int main(int argc, char *argv[])
{
    bool parameterData;
    EnumWindows([](HWND windowHandle, LPARAM parameter) -> BOOL {
        return 1;
    }, reinterpret_cast<LPARAM>(&parameterData));
}

GCC 抛出以下错误消息,我无法理解:

error: invalid user-defined conversion from 'main(int, char**)::<lambda(HWND, LPARAM)>' to 'WNDENUMPROC {aka int (__attribute__((__stdcall__)) *)(HWND__*, long int)}' [-fpermissive]
  }, reinterpret_cast<LPARAM>(&parameterData));
                                             ^
note: candidate is: main(int, char**)::<lambda(HWND, LPARAM)>::operator BOOL (*)(HWND, LPARAM)() const <near match>
  EnumWindows([](HWND windowHandle, LPARAM parameter) -> BOOL {
                                                         ^
note:   no known conversion from 'BOOL (*)(HWND, LPARAM) {aka int (*)(HWND__*, long int)}' to 'WNDENUMPROC {aka int (__attribute__((__stdcall__)) *)(HWND__*, long int)}'
In file included from C:/Qt/Tools/mingw530_32/i686-w64-mingw32/include/Windows.h:72:0,
                 from main.cpp:1:
note:   initializing argument 1 of 'WINBOOL EnumWindows(WNDENUMPROC, LPARAM)'
   WINUSERAPI WINBOOL WINAPI EnumWindows(WNDENUMPROC lpEnumFunc,LPARAM lParam);

海湾合作委员会不喜欢的到底是什么?为了使用 lambda 作为第一个参数来EnumWindows,需要更改什么?

EnumWindows期望使用stdcall调用约定进行回调。GCC 不支持将无状态 lambda 转换为stdcall函数指针,而且我知道无法将 lambda 标记为这样。但是,MSVC 提供对具有各种调用约定的函数指针的转换。

不幸的是,您将不得不使用单独的函数或使用 MSVC:

BOOL CALLBACK callback(HWND windowHandle, LPARAM parameter) { ... }
...
EnumWindows(callback, reinterpret_cast<LPARAM>(&parameterData));
EnumWindows([](HWND windowHandle, LPARAM parameter) -> BOOL __attribute__((__stdcall__)) {
    return 1;