Передача литерала массива в качестве аргумента макроса - PullRequest
12 голосов
/ 31 марта 2011

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

// find the length of an array
#define ARRAY_LENGTH(arr) (sizeof(arr)/sizeof(int))   
// declare an array together with a variable containing the array's 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}.Есть ли способ передать литерал массива в макросы?

РЕДАКТИРОВАТЬ: В некоторых из более сложных макросов, которые мне нужны, мне также может понадобиться передать два или более массивов в макрос,так что не работает variadic макрос.

1 Ответ

26 голосов
/ 31 марта 2011

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

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

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

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

...