Вставка токена в C - PullRequest
       60

Вставка токена в C

6 голосов
/ 22 сентября 2010

После прочтения о VA_NARG

Я пытался реализовать перегрузку функций в зависимости от количества аргументов в C , используя макросы. Теперь проблема:

void hello1(char *s) { ... }
void hello2(char *s, char *t) { ... }
// PP_NARG(...)           macro returns number of arguments :ref to link above
 // does not work
#define hello(...)         hello ## PP_NARG(__VA_ARGS__)  

int main(void)
{
   hello("hi");   // call hello1("hi");
   hello("foo","bar"); // call hello2("foo","bar");
   return 0;
}

Я прочитал это из C-faq. Но все равно не смог заставить его работать ...

Ответы [ 3 ]

4 голосов
/ 22 сентября 2010

Это из-за правил оценки для макросов. Вы должны определить какой-то вспомогательный макрос, который получает число в виде токена:

#define HELLO_1(N, ...)         hello ## N
#define HELLO_0(N, ...)         HELLO_1(N, __VARGS__)
#define HELLO(...)         HELLO_0(PP_NARG(__VA_ARGS__), __VARGS__)  

или около того. Вы также можете заглянуть в предварительный выпуск документации P99 . Это предоставит вам более удобные инструменты макросов, чтобы сделать это напрямую.

4 голосов
/ 22 сентября 2010

Это PP_NARG довольно впечатляющий кусок сумасшествия!

Следуя примеру glue в стандарте C99 (6.10.3.5, пример 4), следующие результаты дают желаемые результаты:

#define glue(a, b)   a ## b
#define xglue(a, b)  glue(a, b)

#define hello(...)   xglue(hello, PP_NARG(__VA_ARGS__))(__VA_ARGS__)
0 голосов
/ 22 сентября 2010

У меня нет доступного для проверки компилятора C99, но это должно работать:

#define helloN(N, ...) hello ## N (__VA_ARGS__)
#define hello(...) helloN(PP_NARG(__VA_ARGS__), __VA_ARGS__)
...