Как printf обрабатывает свои аргументы?

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

5 ответов


такая функция называется вариативную функцию. Вы можете объявить один в 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.


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


эта функция называется переменные числа аргументов в функции. Вы должны включить включаемом файле stdarg.h заголовок файла, а затем использовать va_list тип и 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

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


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

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

разрешить макросы с переменным числом аргументов