编写一个采用可变参数的C宏,并返回一个值

时间:2015-11-25 05:57:59

标签: c macros return-value variadic-functions variadic-macros

我可以使用宏在stderr上显示一条消息,显示当前文件名(来自__FILE__)和行号(来自__LINE__),但也允许使用可选格式和variadic参数列表,用于自定义消息。如果指定了自定义消息,则编译器必须对不正确的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__是可以接受的),而我找到了一些与返回值相关的有用资源。宏(“函数”宏),我无法找到适用于此类可变参数宏的解决方案。

或者,有没有办法检测宏是否已与return语句一起使用并自动停止构建?这个宏似乎没有触发无法访问代码的编译器警告,我不确定为什么会这样。

1 个答案:

答案 0 :(得分:1)

您可以通过不使用@Override protected void onDestroy() { if (BTAdapter.isEnabled()) { BTAdapter.disable(); } this.unregisterReceiver(receiver); super.onDestroy(); } 宏中的语句,仅使用表达式来解决此问题。这意味着你不能声明变量,但必须按原样将所有表达式传递给if (id == R.id.nav_devices) { intent = new Intent("com.navigationwithrssi.RSSIActivity"); startActivity(intent); } 函数,而不是使用临时变量。

也许是这样的

Log2

如果您在返回语句中使用它,例如

DoLog

宏将扩展为

#define Log2(code, format, ...)                                          \
    (DoLog("[%s:%d] %s: " format "\n",                                   \
         strrchr(__FILE__, '/') ? strrchr(__FILE__, '/') + 1 : __FILE__, \
         __LINE__,                                                       \
         ErrorCode_ToString(code),                                       \
         ##__VA_ARGS__                                                   \
     ),                                                                  \
     code /* Will be the result of the expression */                     \
    )

将评估return LogError(ErrorCode_TypeMismatch); 调用,并且由于逗号表达式返回错误代码。