Передача массива в качестве аргумента макроса

Это некоторое время прослушивало меня, например, если я пытаюсь написать этот код:

// find the length of an array
#define ARRAY_LENGTH(arr) (sizeof(arr)/sizeof(int))   
// declare an array together with a variable containing the array length
#define ARRAY(name, arr) int name[] = arr; size_t name##_length = ARRAY_LENGTH(name);

int main() {
    ARRAY(myarr, {1, 2, 3});
}

Код дает эту ошибку:

<stdin>:8:31: error: macro "ARRAY" passed 4 arguments, but takes just 2

Поскольку он видит ARRAY(myarr, {1, 2, 3}); как передающий ARRAY аргумент myarr, {1, 2 и 3}. Есть ли способ передать литерал массива макросам?

EDIT: В некоторых более сложных макросах, которые мне нужны, мне также может потребоваться передать два или более массивов в макрос, поэтому переменный макрос не работает.

Ответ 1

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

#define P99_PROTECT(...) __VA_ARGS__ 
ARRAY(myarr, P99_PROTECT({1, 2, 3}));

Должно работать в вашем случае. Таким образом, у вас есть первый уровень (), который защищает , от интерпретации как разделителя аргументов. Эти () макровызов затем исчезают при расширении.

См. здесь более сложные макросы, которые выполняют развертывание заявления.