【发布时间】:2016-02-27 19:48:11
【问题描述】:
我有一个宏的用途,它在 stderr 上显示一条消息,显示当前文件名(来自__FILE__)和行号(来自__LINE__),但也允许使用可选格式和可变参数列表,用于自定义消息。如果指定了自定义消息,编译器必须对不正确的 printf 参数进行相当不错的警告。
它可以通过以下方式调用,带有这个任意错误代码:
LogError(ErrorCode_TypeMismatch); // prints "[filename:lineno] Type Mismatch:"
LogError(ErrorCode_TypeMismatch, "type was %d", type); // prints "[filename:lineno] Type Mismatch: type was 0"
打印的文件名始终是基本名称(即路径被剥离)。
另外,它返回传入的错误码值,这样就可以按如下方式使用:
result = LogError(ErrorCode_TypeMismatch); // result will be ErrorCode_TypeMismatch
result = LogError(ErrorCode_InvalidName, "name is %s", name); // result will be ErrorCode_InvalidName
目前,我有这个解决方案:
#include <stdio.h>
#include <stdarg.h>
// used to make strings from numeric macros
#define XSTRINGY(a) STRINGY(a)
#define STRINGY(a) #a
typedef enum {
ErrorCode_TypeMismatch,
ErrorCode_InvalidName,
// ...
} ErrorCode;
#define LogError(code, ...) Log2(code, __VA_ARGS__)
// a function to simply return the value provided, used below
static inline ErrorCode ReturnErrorCode(ErrorCode code) { return code; }
#define Log2(code, format, ...) \
ReturnErrorCode(code); /* return the code */ \
{ \
const char * fileName = __FILE__; \
const char * lineNum = XSTRINGY(__LINE__); \
const char * shortFileName = strrchr(fileName, '/'); \
const char * errorString = ErrorCode_ToString(code); /* returns string representation of error code, implementation not shown here */ \
DoLog("[%s:%s] %s: " format "\n", shortFileName ? shortFileName + 1 : fileName, lineNum, errorString, ##__VA_ARGS__); \
}
void DoLog(const char * format, ...) __attribute__ ((format (printf, 1, 2)));
void DoLog(const char * format, ...)
{
va_list argp;
va_start(argp, format);
vfprintf(stderr, format, argp);
va_end(argp);
}
这很好用,并且满足前面提到的要求。
不过我想介绍一个新要求——LogError 宏可以与return 语句一起使用,例如:
return LogError(ErrorCode_TypeMismatch);
// or
return LogError(ErrorCode_TypeMismatch, "The values are %d, %d", value0, value1);
不幸的是,由于 Log2 宏的编写方式(返回传入的值),处理器将执行返回,其余用于格式化文件名/行号并打印的代码将不会执行。因此不会生成任何输出(尽管返回了正确的值)。
我研究了许多关于逗号运算符和 GNU 语句表达式的想法(出于可移植性的原因,我想避免这些想法,尽管##__VA_ARGS__ 是可以接受的)并且我发现了一些与从宏返回值相关的有用资源(“函数”宏),我无法找到适用于此类 variadic 宏的解决方案。
或者,有没有办法检测宏已与return 语句一起使用并自动停止构建?无法访问代码的编译器警告似乎不会通过此宏触发,我不确定这是为什么。
【问题讨论】:
-
只是一个不相关的问题,但是为什么将行 (
__LINE__) 打印为 字符串 而不是使用"%d"格式的十进制数? -
另外,您当前的
Log2宏存在严重缺陷,如果它是不带大括号的if语句体的一部分,例如:if (some_condition) LogError(...);只有ReturnErrorCode调用(这里真的是无操作)才会处于这种状态。 -
抛弃变量,直接在 DoLog 参数中使用 __FILE__ 等。
-
@JoachimPileborg 是的,我认为这个缺陷是问题的一部分,为什么
return LogError(...)也是有问题的。将__LINE__转换为字符串是有原因的,但它在时间上丢失了。我可以看到用%d打印它就足够了。 -
@n.m.我确实有这个变化的变体,你是对的,这是一个合理的简化,谢谢。但是它并没有解决返回值问题。
标签: c macros return-value variadic-functions variadic-macros