Макрос с использованием переменных аргументов - PullRequest
2 голосов
/ 14 января 2020

Пожалуйста, дайте мне знать, если есть способ, где я мог бы использовать переменные аргументы внутри макрофункции и заставить работающий код макроса ниже.

Другими словами, есть ли способ, где я мог бы использовать локальный переменные в va_start.

typedef struct
{
    char *p1;
    char *p2;
    char *p3;
}Pointers;

#define NULLCHECK_VR(num, ...)                          \
        {                                               \
            va_list valist;                             \
            int     v_index;                            \
                                                        \
            va_start(valist, num);                      \
                                                        \
            for( v_index=0; v_index < num; v_index++)   \
            {                                           \
                if( NULL == va_arg(valist, void *))     \
                {                                       \
                    printf("NULL \n");                  \
                }                                       \
                else                                    \
                {                                       \
                    printf("NOT NULL \n");              \
                }                                       \
            }                                           \
                                                        \
            va_end(valist);                             \
        }

int main(void)
{
    Pointers ptr;

    memset(&ptr, 0, sizeof(Pointers));

    NULLCHECK_VR(3, ptr.p1, ptr.p2, ptr.p3)

    return (0);
}

Ответы [ 2 ]

6 голосов
/ 14 января 2020

Вы можете использовать обычную функцию:

void NULLCHECK_VR(int num, ...)
{
    va_list valist;
    int     v_index;

    va_start(valist, num);

    for( v_index=0; v_index < num; v_index++)
    {
        if( NULL == va_arg(valist, void *))
        {
            printf("NULL\n");
        }
        else
        {
            printf("NOT NULL\n");
        }
    }

    va_end(valist);
}

или в C ++ (17)

template <typename Ts>
void NULLCHECK_VR(const Ts*... ptrs)
{
    ((std::cout << (ptrs == nullptr ? "NULL\n" : "NOT NULL\n")), ...);
}
2 голосов
/ 14 января 2020

Вы не можете использовать va_list для доступа к локальным переменным. Макросы va_start, va_copy и va_end предназначены только для обработки аргументов функции variadi c. Макрос не является функцией.

Для макросов C variadi c идентификатор __VA_ARGS__ в тексте замены макроса будет расширен до аргументов c variadi, поэтому возможная реализация вашего * Макрос 1008 * выглядит следующим образом:

#define NULLCHECK_VR(num, ...)                          \
    do {                                                \
        void *_ptrs[] = { __VA_ARGS__ };                \
        int _num = (num);                               \
        int v_index;                                    \
                                                        \
        for (v_index = 0; v_index < _num; v_index++)    \
        {                                               \
            if (NULL == _ptrs[v_index])                 \
            {                                           \
                printf("NULL \n");                      \
            }                                           \
            else                                        \
            {                                           \
                printf("NOT NULL \n");                  \
            }                                           \
        }                                               \
    } while (0)

Здесь __VA_ARGS__ используется для инициализации локального _ptrs массива.

(я добавил локальную переменную _num, чтобы избежать расширения параметр num несколько раз, и я использовал идиому do while (0), чтобы превратить блок в оператор.)

...