Есть ли лучший способ сделать с именованными аргументами? - PullRequest
6 голосов
/ 10 мая 2011

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

Есть ли лучший способ использовать именованные аргументы в c, чем следующий?

enum named_args {NAME,ADRESS,AGE,NA_SENTINEL};

void named_arg_initializers(struct person p, enum * named_args)
{
    enum named_args * current_name;
    enum named_args * current_arg;
    ...
    if(named_args==NULL)
        return;
    current_name = named_args[0];
    current_arg = named_args[1];
    while(current_name!=NA_SENTINEL)
    {
        current_name+=2;
        current_arg+=2;
        if(current_name==NAME)
            p.name=current_arg;
        else if(...
        ...
        }
        ...
    }
    ...
}

Ответы [ 2 ]

18 голосов
/ 10 мая 2011

Конечно.

struct toto {
  unsigned age;
  char name[25];
};

void func(struct toto);

...

func((struct toto){ .name = "you", .age = 18 });

или, если хотите, можете обернуть это в макрос

#define FUNC(...) func((struct toto){ __VA_ARGS__ })
...
FUNC(.name = "you", .age = 18 );
1 голос
/ 10 мая 2011

Способ, который вы показали, недопустим, если все именованные аргументы не совместимы с enum (это можно исправить с помощью аргумента void *).

Однако вы можете сделать что-топохож на varargs, который выглядит аккуратнее:

#include <stdarg.h>

enum named_args { NAME, ADDRESS, AGE, NA_SENTINEL };

void named_arg_initializers(struct person *p, ...)
{
    va_list ap;
    enum named_args argn;

    va_start(ap, p);
    for (argn = va_arg(ap, enum named_args); argn != NA_SENTINEL; argn = va_arg(ap, enum named_args))
    {
        switch (argn)
        {
          case NAME:
            p->name = va_arg(ap, char *);
            break;

          case AGE:
            p->age = va_arg(ap, int);
            break;

          /* ...  */
         }
    }
    va_end(ap);

    /* ... */
}

Вы бы использовали его так:

named_arg_initializers(&p, AGE, 110, NAME, "Claude Choules", NA_SENTINEL);
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...