0

getopt_long()初めて関数を使用しようとしていますが、フラグではない引数に問題があります。たとえば、私のコードでは、不明な引数が与えられたときに、それを入力ファイルとして使用したいと考えています。ファイル名のみでこれを実行すると、印刷されません。最初にフラグ、任意のフラグを使用すると、印刷できます。

どうすればこれを修正できますか?

 #include <stdio.h>
 #include <getopt.h>

 static struct option long_options[] = {
  {"help",  no_argument,       0, 'h'},
  {"input",  required_argument, 0, 'i'},
  {"output",  required_argument, 0, 'o'},
  {"algorithm",  required_argument, 0, 'a'},
  {0, 0, 0, 0}
 };

 int main(int argc, char *argv[]) {
  int c;
  int option_index = 0;

  while(42) {
   c = getopt_long(argc, argv, "hi:o:a:", long_options, 
     &option_index);
   if(c == -1)
    break;

   switch(c) {
    case 'h': /* --help */
     printf("--help flag\n");
     break;
    case 'i': /* --input */
     printf("--input flag\n");
     break;
    case 'o': /* --output */
     printf("--output flag\n");
     break;
    case 'a': /* --algorithm */
     printf("--algorithm flag \n");
     break;
    default: /* ??? */
     fprintf(stderr, "Invalid option");
     return 1;
   }

   if(optind < argc) {
    printf("other arguments: ");

    while(optind < argc) {
     printf ("%s ", argv[optind]);
     optind++;
    }

    printf("\n");
   }
  }

  return 0;
 }
4

2 に答える 2

5

ループにはスイッチのみを含める必要があります。別の (ネストされていない) ループで残りの引数を処理します。

#include <stdio.h>
#include <getopt.h>

static struct option long_options[] =
{
    {"help",  no_argument,       0, 'h'},
    {"input",  required_argument, 0, 'i'},
    {"output",  required_argument, 0, 'o'},
    {"algorithm",  required_argument, 0, 'a'},
    {0, 0, 0, 0}
};

int main(int argc, char *argv[])
{
    int opt;
    int option_index = 0;
    int i;

    while ((opt = getopt_long(argc, argv, "hi:o:a:", long_options, &option_index)) != -1)
    {
        switch(opt)
        {
            case 'h': /* --help */
                printf("--help flag\n");
                break;
            case 'i': /* --input */
                printf("--input flag (%s)\n", optarg);
                break;
            case 'o': /* --output */
                printf("--output flag (%s)\n", optarg);
                break;
            case 'a': /* --algorithm */
                printf("--algorithm flag (%s)\n", optarg);
                break;
            default: /* ??? */
                fprintf(stderr, "Invalid option %c\n", opt);
                return 1;
        }
    }

    for (i = optind; i < argc; i++)
        printf("Process: %s\n", argv[i]);

    return 0;
}

'文字' ^A '\1'; を使用したオプションであるかのように、GNU を使用してgetopt()ファイル名の引数を返す方法があります。短いオプション文字列の最初の文字としてgetopt_long()' ' を使用し、スイッチで ' ' をトラップします。の値はファイルの名前です。-\1optarg

#include <stdio.h>
#include <getopt.h>

static struct option long_options[] =
{
    {"help",  no_argument,       0, 'h'},
    {"input",  required_argument, 0, 'i'},
    {"output",  required_argument, 0, 'o'},
    {"algorithm",  required_argument, 0, 'a'},
    {0, 0, 0, 0}
};

int main(int argc, char *argv[])
{
    int opt;
    int option_index = 0;
    int i;

    while ((opt = getopt_long(argc, argv, "-hi:o:a:", long_options, &option_index)) != -1)
    {
        switch(opt)
        {
            case 'h': /* --help */
                printf("--help flag\n");
                break;
            case 'i': /* --input */
                printf("--input flag (%s)\n", optarg);
                break;
            case 'o': /* --output */
                printf("--output flag (%s)\n", optarg);
                break;
            case 'a': /* --algorithm */
                printf("--algorithm flag (%s)\n", optarg);
                break;
            case '\1':
                printf("File: %s\n", optarg);
                break;
            default: /* ??? */
                fprintf(stderr, "Invalid option %c\n", opt);
                return 1;
        }
    }

    for (i = optind; i < argc; i++)
        printf("Process: %s\n", argv[i]);

    return 0;
}

ただし、意地悪なユーザーが次のように入力した場合に備えて、引数処理ループの後にループが必要です。

program -- abc def

' 'は、ファイル名の引数を処理せずにループを--終了します。while()

于 2010-09-09T20:41:04.277 に答える
3

while-1 を返すときに外側のループから抜け出すため、機能しませんgetopt_long(これは、これ以上オプションがないことを示します)。

if (optind < argc)ブロックを外側のwhileループの外に移動する必要があります。とにかくそこに属していません。while外側のループを次のように記述した場合、これはより明確になる可能性があります。

while ((c = getopt_long(...)) != -1)
{
    switch (c)
    {
        /* Deal with flags. */
    }
}
于 2010-09-09T20:41:52.513 に答える