尝试将结构成员函数传递给 atexit

Trying to pass a struct member function to atexit

我正在用 C++ 编写调试器,退出时我需要它来清除在调试对象中设置的所有断点。

我制作了一个名为 BREAKPOINT_INFOstruct,其中包含清除每个断点所需的信息。进程句柄(hProcess)、断点的虚拟内存地址(lpBreakPoint)、被替换的原指令的操作码(instr)等信息。

然后我有一个 CleanUp 成员函数,它使用上述数据成员来恢复原始指令,这样调试对象就不会崩溃。

typedef struct
{
    HANDLE hProcess;

    PCHAR  lpBreakPoint;

    CHAR   instr;

    void CleanUp()
    {
        DWORD dwError = 0;
        std::ostringstream oss;
        LPSTR szErrorRest = (LPSTR)"Error restoring original instruction: ";

        if(!WriteProcessMemory(hProcess, lpBreakPoint, &instr, sizeof(CHAR), NULL))
        {
            dwError = GetLastError();

            oss << szErrorRest << dwError;

            szErrorRest = oss.str().c_str();

            MessageBox(NULL, szErrorRest, "ERROR", MB_OK|MB_ICONERROR);
        }
    }
}BREAKPOINT_INFO, *PBREAKPOINT_INFO;

但是,当我尝试为每个 BREAKPOINT_INFO 结构传递我的成员函数时,出现编译错误:

C:\...\...\main.cpp|39|error: cannot convert 'BREAKPOINT_INFO::CleanUp' from type 'void (BREAKPOINT_INFO::)()' to type 'void (__attribute__((__cdecl__)) *)()'|

即使在尝试转换为该类型之后,它也无法编译。是否可以将成员函数传递给 atexit?如果是这样,我将如何去做?谢谢你的时间。

不可以,除了声明中指定的 atexit 函数指针外,您不能传递任何其他内容。

int atexit (void (*func)(void)) noexcept

成员函数不是自由函数。
正如您在错误消息中看到的那样,它的类型 void (BREAKPOINT_INFO::)()void ().

not 相同

实际上,指向成员函数的指针采用额外的第一个参数来指定 class 的哪个实例正在调用成员函数。

auto member_function_pointer = BREAKPOINT_INFO::CleanUp;
BREAKPOINT_INFO some_info;
member_function_pointer(&some_info);  //is equivalent to...
some_info.CleanUp();

不,您不能将捕获的 lambda 作为函数指针传递。非捕获 lambda 有一个 ,这可能是函数指针可以指向其他东西的误解的来源。

如评论中所述,您应该改为使用析构函数进行清理工作。这种模式被称为 RAII,是迄今为止最好的模式之一。


作为旁注,您不需要在 C++ 中对 struct 声明做疯狂的 typedef,您正在编写的内容看起来很像 C 语言的方法。

我原本不打算写一个答案,因为即使正确和真实,它也会邀请 - 投票,但无论如何,就这样吧。

如果你cify it first, here's an example:

,你肯定可以将捕获的 lambda 对象传递给 atexit()
int a = 10;

std::atexit(cify<void(*)()>([a](){std::cout << a << std::endl;}));

但是,为了解决您的问题,我宁愿使用某种 scope exit 方案。

如果您捕获了某个 class 实例的指针或引用,您将能够访问其成员。