Подсчет количества параметров в вызове метода аргумента переменной C - PullRequest
6 голосов
/ 06 октября 2010

При использовании va_start (), va_arg () и va_end () для чтения параметров, переданных методу, есть ли способ подсчитать, сколько существует аргументов?

Согласно справочной странице, если вы слишком часто вызываете va_arg (), вы получаете «случайные ошибки»:

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

Ответы [ 3 ]

11 голосов
/ 06 октября 2010

Нет. Функция аргумента переменной (например, printf) должна «знать», когда прекратить искать дополнительные аргументы.

printf знает по номеру %d, %s и другим символам в строке формата.

Другие функции иногда используют значения Sentinel:

sumValues(1, 3, 5, 7, 6, 9, -1); // will add numbers until it encounters a -1

Другие функции могут иметь количество параметров, указанных заранее:

AddNames(4, "Bill", "Alice", "Mike", "Tom");
8 голосов
/ 06 октября 2010

Вот довольно удивительный трюк, который позволяет вам строить то, что вы хотите, используя переменные макросы C99:

PP_NARG ()
Возвращает количество аргументов, содержащихся в__VA_ARGS__

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

См. Также этот вопрос.Однако, вероятно, классические подходы еще более разумны в течение года или трех!


Код, так что этот ответ полезен даже без архива Usenet ...

/* The PP_NARG macro returns the number of arguments that have been
  * passed to it.
  */

#define PP_NARG(...) \
         PP_NARG_(__VA_ARGS__,PP_RSEQ_N())
#define PP_NARG_(...) \
         PP_ARG_N(__VA_ARGS__)
#define PP_ARG_N( \
          _1, _2, _3, _4, _5, _6, _7, _8, _9,_10, \
         _11,_12,_13,_14,_15,_16,_17,_18,_19,_20, \
         _21,_22,_23,_24,_25,_26,_27,_28,_29,_30, \
         _31,_32,_33,_34,_35,_36,_37,_38,_39,_40, \
         _41,_42,_43,_44,_45,_46,_47,_48,_49,_50, \
         _51,_52,_53,_54,_55,_56,_57,_58,_59,_60, \
         _61,_62,_63,N,...) N
#define PP_RSEQ_N() \
         63,62,61,60,                   \
         59,58,57,56,55,54,53,52,51,50, \
         49,48,47,46,45,44,43,42,41,40, \
         39,38,37,36,35,34,33,32,31,30, \
         29,28,27,26,25,24,23,22,21,20, \
         19,18,17,16,15,14,13,12,11,10, \
         9,8,7,6,5,4,3,2,1,0 
5 голосов
/ 06 октября 2010

Есть два способа узнать, сколько аргументов было передано.Во-первых, вы можете указать один из параметров функции (например, printf).Другой способ заключается в том, чтобы в конце списка было указано значение часового - например, вы можете остановиться на аргументе NULL.

. Вы можете использовать va_copy, если выберете второй метод, но все жехочу посчитать параметры, прежде чем решить, что с ними делать.

...