1

1 つまたは 2 つのパラメーターを受け入れるマクロを定義したいと考えています。両方のパラメーターは異なるタイプである必要があります。省略記号を使用して渡された引数を読み取る方法は?

以下にサンプルを示します。

void test(char *var2) 
{ 
 printf("%s\n",var2); 
} 

#define PRINT_STRING(...) ( if (!var1) test(var2) ) 

int main(int argc, _TCHAR argv[]) {

PRINT_STRING(TRUE); 
PRINT_STRING(FALSE,"Hello, World!"); 
return 0;
} 
4

4 に答える 4

4

これはVariadic マクロとして知られています。

于 2012-07-16T23:49:22.150 に答える
2

コンパイラがをサポートしている場合は__VA_ARGS__、次のように実行できます。

#include <stdio.h>

#define NUM_ARGS__(X, \
                      N64,N63,N62,N61,N60, \
  N59,N58,N57,N56,N55,N54,N53,N52,N51,N50, \
  N49,N48,N47,N46,N45,N44,N43,N42,N41,N40, \
  N39,N38,N37,N36,N35,N34,N33,N32,N31,N30, \
  N29,N28,N27,N26,N25,N24,N23,N22,N21,N20, \
  N19,N18,N17,N16,N15,N14,N13,N12,N11,N10, \
  N09,N08,N07,N06,N05,N04,N03,N02,N01,  N, ...) N

#define NUM_ARGS(...) \
  NUM_ARGS__(0, __VA_ARGS__, \
                 64,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)

#define PRINT_STRING_1(var) \
  { if (!(var)) {} }

#define PRINT_STRING_2(var, ...) \
  { if (!(var)) test(__VA_ARGS__); }

#define PRINT_STRINGN__(N, ...) \
  PRINT_STRING_##N(__VA_ARGS__)

#define PRINT_STRINGN(N, ...) \
  PRINT_STRINGN__(N, __VA_ARGS__)

#define PRINT_STRING(...) \
  PRINT_STRINGN(NUM_ARGS(__VA_ARGS__), __VA_ARGS__) 

void test(char* var2) 
{ 
  printf("%s\n", var2); 
} 

int main(void)
{
  PRINT_STRING(1);
  PRINT_STRING(0, "Hello, World!");
  PRINT_STRING(1, "You can't see me!");
  return 0;
}

出力:

Hello, World!
于 2012-07-17T06:56:51.407 に答える
2

そのようなことを行うには、次のような一連のマクロを実装する必要があります

#include <stdbool.h>
#define PRINT_STRING0(X, Y) do { if (X && Y) test(Y); } while(false) 
#define PRINT_STRING1(X, Y, ...) PRINT_STRING0(X, Y)
#define PRINT_STRING(...) PRINT_STRING1(__VA_ARGS__, 0, 0)

それらの最後 (ユーザー インターフェイス) は、 の 2 番目または 3 番目の引数を追加します0PRINT_STRING1より大きいすべての引数を無視します2。そしてPRINT_STRING0、仕事をします。

いくつかのより多くの発言:

  • ここでプログラムしたいようなマクロは、他のステートメントと同じように構文的に配置できることに注意してください。ここの例では、do { } while(false)そのトリックを行います
  • C99 にはブール型と定数があるため、これらは_Boolorboolfalseandです。true
于 2012-07-17T07:02:07.623 に答える
0

そのためにマクロを使用しないでください。可変引数関数を使用してください。

void print_string( bool should_print, ... )
{
    if( should_print )
    {
        va_list argp;
        va_start( argp, should_print);
        char *string = va_arg(argp, char *);
        if( string) printf("%s", string );
        va_end( argp );

    }
}

しかし、va_arg は本当に 2 番目の引数があるかどうかをチェックしないため、これらの種類のものを使用するときは十分に注意してください。

また、1 つまたは 2 つの引数を受け取るマクロを使用するには、GCC のトリック (このページの一番下にあります)を使用します。

#define PRINT_STRING( should , args... ) print_string( should , ##args )

(コンマ間のスペースに注意してください)

于 2012-07-17T06:36:28.850 に答える