Как printf обрабатывает свои аргументы? - PullRequest
15 голосов
/ 12 марта 2010

Как printf обрабатывает свои аргументы? Я знаю, что в C # я могу использовать ключевое слово params, чтобы сделать что-то подобное, но я не могу сделать это в C?

Ответы [ 5 ]

18 голосов
/ 12 марта 2010

Такая функция называется переменная функция . Вы можете объявить один в C, используя ..., например:

int f(int, ... );

Затем вы можете использовать va_start, va_arg и va_end для работы со списком аргументов. Вот пример:

#include <stdlib.h>
#include <stdarg.h>
#include <stdio.h>

void f(void);

main(){
        f();
}

int maxof(int n_args, ...){
        register int i;
        int max, a;
        va_list ap;

        va_start(ap, n_args);
        max = va_arg(ap, int);
        for(i = 2; i <= n_args; i++) {
            if((a = va_arg(ap, int)) > max)
                max = a;
        }

        va_end(ap);
        return max;
}

void f(void) {
        int i = 5;
        int j[256];
        j[42] = 24;
        printf("%d\n",maxof(3, i, j[42], 0));
}

Для получения дополнительной информации см. Книга C и stdarg.h .

9 голосов
/ 12 марта 2010

Эта функция называется Переменные числа аргументов в функции . Вы должны включить заголовочный файл stdarg.h ; затем используйте va_list type и va_start , va_arg и va_end в теле вашей функции:

void print_arguments(int number_of_arguments, ...)
{
  va_list list;
  va_start(list, number_of_arguments);
  printf("I am first element of the list: %d \n", va_arg(list, int));
  printf("I am second element of the list: %d \n", va_arg(list, int));
  printf("I am third element of the list: %d \n", va_arg(list, int));
  va_end(list);
}

Затем вызовите свою функцию так:

print_arguments(3,1,2,3);

, который распечатает следующее:

    I am first element of the list: 1
    I am second element of the list: 2
    I am third element of the list: 3
8 голосов
/ 12 марта 2010

Способ, которым это делается в C, называется "varargs". Для этого есть учебник: http://c -faq.com / ~ scs / cclass / int / sx11b.html

7 голосов
/ 12 марта 2010

Как уже говорили другие, printf использует va_args для работы. Это довольно крутое упражнение - написать собственную версию printf, если не что иное, чтобы проверить, что printf, в отличие от writeln Паскаля, не волшебство компилятора. После того, как вы это сделаете, вы должны уйти от этого. Вот статья в блоге Я написал подробно, почему (краткий ответ: вы можете создавать ошибки, которые могут долго оставаться незамеченными).

5 голосов
/ 25 сентября 2010

и просто для завершения истории gcc (не уверен насчет других компиляторов) поддерживает

#define FUNC(X,Y,...) wiz(X,Y, ##__VA_ARGS__)

чтобы разрешить вариационные макросы

...