尝试将结构成员函数传递给 atexit
Trying to pass a struct member function to atexit
我正在用 C++ 编写调试器,退出时我需要它来清除在调试对象中设置的所有断点。
我制作了一个名为 BREAKPOINT_INFO
的 struct
,其中包含清除每个断点所需的信息。进程句柄(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 实例的指针或引用,您将能够访问其成员。
我正在用 C++ 编写调试器,退出时我需要它来清除在调试对象中设置的所有断点。
我制作了一个名为 BREAKPOINT_INFO
的 struct
,其中包含清除每个断点所需的信息。进程句柄(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 ()
.
实际上,指向成员函数的指针采用额外的第一个参数来指定 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 实例的指针或引用,您将能够访问其成员。