Truco macro variádico

3 minutos de lectura

avatar de usuario
R.. GitHub DEJAR DE AYUDAR A ICE

¿Cuál es el truco para crear una macro variádica? FOO(a1, a2, a3,..., an) tal que se expande a FOOn(a1, a2, a3,..., an) para valores de n en cualquier rango acotado preseleccionado que elija? Es decir, FOO(a) debería expandirse a FOO1(a), FOO(a, b, c) para FOO3(a, b, c)etc. Sé que hay un truco estándar, pero parece que no puedo encontrarlo.

Siéntase libre de marcar esta pregunta como duplicada y cerrarla si hay otra pregunta con la respuesta. Sospecho que hay pero no pude encontrarlo.

  • esta publicación tiene algo que puede resultarle útil: stackoverflow.com/questions/3420459/…

    – Estera

    19/03/2011 a las 23:00

  • vea la respuesta aquí: stackoverflow.com/questions/2124339/…

    – Cualquier grano

    19 de marzo de 2011 a las 23:14

  • @aaa: Las respuestas a esa pregunta no responden a esta pregunta. La pregunta vinculada por @Mat es más similar, aunque no la consideraría un duplicado exacto.

    –James McNellis

    20 de marzo de 2011 a las 0:26

#define VA_NARGS_IMPL(_1, _2, _3, _4, _5, N, ...) N
#define VA_NARGS(...) VA_NARGS_IMPL(__VA_ARGS__, 5, 4, 3, 2, 1)

#define FOO_IMPL2(count, ...) FOO ## count (__VA_ARGS__)
#define FOO_IMPL(count, ...) FOO_IMPL2(count, __VA_ARGS__) 
#define FOO(...) FOO_IMPL(VA_NARGS(__VA_ARGS__), __VA_ARGS__)

FOO(a)
FOO(a, b)
FOO(a, b, c)

Las invocaciones se sustituyen por:

FOO1 (a)
FOO2 (a, b)
FOO3 (a, b, c)

  • Excelente respuesta, pero ¿cómo resuelve el caso de parámetro cero al llamar? FOO()donde debería expandirse a FOO0 ()pero su solución se expande a FOO1 ()? Me gustaría saber cómo hacer esto en c99 puro (sin extensión GNU)

    – mchiasson

    5 de junio de 2015 a las 16:48

  • @mchiasson: Hasta donde yo sé, no hay forma de hacerlo en el estándar C. Debe pasar al menos un token al __VA_ARGS__; no puede llamar a una macro de este tipo con una lista de argumentos vacía.

    –James McNellis

    5 de junio de 2015 a las 21:39

avatar de usuario
Estera

Esta publicación Macro variádica para contar el número de argumentos tiene lo que buscas creo. Mira la primera y la segunda respuesta.

  • Gracias, este enlace era justo lo que necesitaba. Es mucho mejor que los posibles duplicados publicados.

    – R.. GitHub DEJA DE AYUDAR A ICE

    20 de marzo de 2011 a las 0:44

Mejorando la respuesta de James para agregar algo de flexibilidad:

#define VA_NARGS_IMPL(_1, _2, _3, _4, _5, N, ...) N
#define VA_NARGS(...) VA_NARGS_IMPL(X,##__VA_ARGS__, 4, 3, 2, 1, 0)
#define VARARG_IMPL2(base, count, ...) base##count(__VA_ARGS__)
#define VARARG_IMPL(base, count, ...) VARARG_IMPL2(base, count, __VA_ARGS__) 
#define VARARG(base, ...) VARARG_IMPL(base, VA_NARGS(__VA_ARGS__), __VA_ARGS__)

#define MyMacro0() Also works without arguments.
#define MyMacro2(x,y) [x...y]
#define MyMacro(...) VARARG(MyMacro, __VA_ARGS__)

MyMacro()
MyMacro(a)
MyMacro(a, b)
MyMacro(a, b, c)

Producción:

Also works without arguments.
MyMacro1(a)
[a...b]
MyMacro3(a, b, c)

  • Excelente respuesta, pero ##__VA_ARGS__ solo funciona con la extensión GNU. ¿Sabes cómo resolver esto para c99?

    – mchiasson

    5 de junio de 2015 a las 16:49

¿Ha sido útil esta solución?

Esta web utiliza cookies propias y de terceros para su correcto funcionamiento y para fines analíticos y para mostrarte publicidad relacionada con sus preferencias en base a un perfil elaborado a partir de tus hábitos de navegación. Al hacer clic en el botón Aceptar, acepta el uso de estas tecnologías y el procesamiento de tus datos para estos propósitos. Configurar y más información
Privacidad