我正在尝试编写一个dbgassert
类似于标准的宏assert
。除了做什么assert
,我想dbgassert
打印任意数量的附加参数(包含调试信息)。
下面列出了我到目前为止所拥有的内容,它改编自这个 SO answer。但是我在代码中遇到了可变参数模板或宏的问题。如果我使用至少一个附加参数(OK 行),dbgassert
则按预期工作。但是如果我没有给出额外的论点,那么编译就会失败(问题行)。
我有一些可变参数模板编程的经验(例如如何打印元组),但我以前没有使用过可变参数宏。
有人可以解释一下编写这个可变参数宏组合的正确方法是什么吗?
顺便说一句,有人能解释一下#EX
宏中的魔法吗?它显示了表达式并在 gcc4.8.1 上为我工作。是否普遍支持?
谢谢,
代码:
//corrected reserved identifier issue and assumption issues per comments
#include <cassert>
#include <iostream>
using namespace std;
template <typename ...Args>
void realdbgassert(const char *msg, const char *file, int line, Args ... args) {
cout << "Assertion failed! \nFile " << file << ", Line " << line << endl
<< " Expression: " << msg << endl;
std::abort();
}
#define dbgassert(EX,...) \
(void)((EX) || (realdbgassert (#EX, __FILE__, __LINE__, __VA_ARGS__),0))
int main() {
dbgassert(1>2,"right","yes"); //OK
dbgassert(1>2,"right"); //OK.
//dbgassert(1>2); //Problem. compile error: expected primary-expression before ')' token
//#define dbgassert(EX,...) (void)((EX) || (realdbgassert (#EX, __FILE__, __LINE__, __VA_ARGS__)^,0))
}
代码的原始版本。
#include <cassert>
#include <sstream>
using namespace std;
#ifdef __cplusplus
extern "C" {
#endif
extern void __assert (const char *msg, const char *file, int line);
#ifdef __cplusplus
};
#endif
template <typename ...Args>
void _realdbgassert(const char *msg, const char *file, int line, Args ... args) {
stringstream os;
//... do something
__assert(msg,file,line);
}
#define dbgassert(EX,...) (void)((EX) || (_realdbgassert (#EX, __FILE__, __LINE__, __VA_ARGS__),0))
int main() {
dbgassert(1==0,"right"); //Problem line: undefined reference to `__assert'
}