getopt_long

getopt_long可以用来解析命令行参数,也可以进行参数解析。例如

#include <stdio.h>     /* for printf */
#include <stdlib.h>    /* for exit */
#include <getopt.h>

static int verbose_flag;

int
parse(int argc, char **argv)
{
    int c;
    int digit_optind = 0;

    while (1) {
        int this_option_optind = optind ? optind : 1;
        int option_index = 0;
        static struct option long_options[] = {
            {"verbose", no_argument,       &verbose_flag, 1},
            {"add",     required_argument, 0,  0 },
            {"append",  no_argument,       0,  0 },
            {"delete",  required_argument, 0,  0 },
            {"verbose", no_argument,       0,  0 },
            {"create",  required_argument, 0, 'c'},
            {"file",    required_argument, 0,  0 },
            {0,         0,                 0,  0 }
        };

       c = getopt_long(argc, argv, "abc:d:012",
                 long_options, &option_index);
        if (c == -1)
            break;

       switch (c) {
        case 0:
            printf("option %s", long_options[option_index].name);
            if (optarg)
                printf(" with arg %s", optarg);
            printf("\n");
            break;

        case '0':
        case '1':
        case '2':
            if (digit_optind != 0 && digit_optind != this_option_optind)
              printf("digits occur in two different argv-elements.\n");
            digit_optind = this_option_optind;
            printf("option %c\n", c);
            break;

       case 'a':
            printf("option a\n");
            break;

       case 'b':
            printf("option b\n");
            break;

       case 'c':
            printf("option c with value '%s'\n", optarg);
            break;

       case 'd':
            printf("option d with value '%s'\n", optarg);
            break;

       case '?':
            break;

       default:
            printf("?? getopt returned character code 0%o ??\n", c);
        }
    }

    if (optind < argc) {
        printf("non-option ARGV-elements: ");
        while (optind < argc)
            printf("%s ", argv[optind++]);
        printf("\n");
    }

    if (verbose_flag)
        printf ("verbose flag is set\n");

    return 0;
}

void
main()
{
        char *argv[6]={0};
        int argc = 6;
        argv[0] = "prog1";
        argv[1] = "--add";
        argv[2] = "AA";
        argv[3] = "-d";
        argv[4] = "BB";
        argv[5] = "--verbose";

        parse(argc, argv);

        optind = 1;

        char *argv2[4]={0};
        int argc2 = 4;
        argv2[0] = "prog2";
        argv2[1] = "--file";
        argv2[2] = "CC";
        argv2[3] = "--verbose";
        //argv2[3] = "--delete";
        //argv2[4] = "DD";

        parse(argc2, argv2);
        return ;
}

 

getopt可以解析一个字母的命令行参数, “-d”, getopt_long则可以解析更长名称的命令行参数, 以”–“开头, 如例子中的”–file”.

参数比getopt多了两个, 函数原型如下:

#include <unistd.h>

int getopt(int argc, char * const argv[],
           const char *optstring);

extern char *optarg;
extern int optind, opterr, optopt;

#include <getopt.h>

int getopt_long(int argc, char * const argv[],
           const char *optstring,
           const struct option *longopts, int *longindex);

int getopt_long_only(int argc, char * const argv[],
           const char *optstring,
           const struct option *longopts, int *longindex);

 

另外注意函数依赖的定义, 编译时可以指定

getopt(): _POSIX_C_SOURCE >= 2 || _XOPEN_SOURCE
getopt_long
(), getopt_long_only(): _GNU_SOURCE

getopt_long 参数结构体相对复杂一些

struct option {
    const char *name;
    int         has_arg;
    int        *flag;
    int         val;
};


static struct option long_options[] = {
    {"verbose", no_argument,       &verbose_flag, 1},
    {"add",     required_argument, 0,  0 },
    {"append",  no_argument,       0,  0 },
    {"delete",  required_argument, 0,  0 },
    {"verbose", no_argument,       0,  0 },
    {"create",  required_argument, 0, 'c'},
    {"file",    required_argument, 0,  0 },
    {0,         0,                 0,  0 }
};

 

如果设置了flag不为null, 那么将val值复制给flag指针指向变量。

例子中是将verbose_flag设置为 1.

如果设置了flag为null, 那么val作为返回值, 这个返回值可以是设置为含义相同的短类型参数的标签,

例子中的c和create是相同的含义, 一个是长类型参数, 一个是短类型的参数。

 

另外还有个比较重要的全局变量:

extern char *optarg; 
extern int optind, opterr, optopt;

如果 getopt() 无法识别选项字符,它将向 stderr 打印一条错误消息,将该字符存储在 optopt 中,并返回“?”。调用程序可以通过将 opterr 设置为 0 来防止出现错误消息。

因为optind存储的下一个要解析的参数, 如果是getopt_long, 可以保留optind位置, 解析出错时显示

#include <stdio.h>     /* for printf */
#include <stdlib.h>    /* for exit */
#include <getopt.h>

static int verbose_flag;

int
parse(int argc, char **argv)
{

       ...

       case '?':
            printf("unkown option  '%s'\n", argv[old_optnd]);
            break;

       default:
            printf("?? getopt returned character code 0%o ??\n", c);
        }
       old_optnd = optind;
    }
    ...
}

void
main()
{
        opterr = 0;
        char *argv[8]={0};
        int argc = 8;
        argv[0] = "prog1";
        argv[1] = "--add";
        argv[2] = "AA";
        argv[3] = "-d";
        argv[4] = "BB";
        argv[5] = "--verbose";
        argv[6] = "--error";
        argv[7] = "ddd";

        parse(argc, argv);

        optind = 1;

        char *argv2[4]={0};
        int argc2 = 4;
        argv2[0] = "prog2";
        argv2[1] = "--file";
        argv2[2] = "CC";
        argv2[3] = "--verbose";
        //argv2[3] = "--delete";
        //argv2[4] = "DD";

        parse(argc2, argv2);
        return ;
}

 

如果不设置 opterr, 会有错误输出 prog1: unrecognized option ‘–error’

garlic@garlic:~/getoption$ ./test
option add with arg AA
option d with value 'BB'
option verbose
prog1: unrecognized option '--error'
unkown option  '--error'
non-option ARGV-elements: ddd
verbose flag is set
option file with arg CC
option verbose
verbose flag is set

opterr设置为1 则没有相关输出。

解析完毕后记得要重新optind, 因为这个变量使用后不会重置。会出现 Segment Fault

 

 

参考及引用

https://www.gnu.org/software/libc/manual/html_node/Getopt-Long-Option-Example.html

https://linux.die.net/man/3/getopt_long

https://cplusplus.com/forum/unices/104742/ optind问题

图片from陳為

 

Comments are closed.