谷动谷力

标题: 【RT-Thread】optparse一个开源, 可移植的, 可重入的和可嵌入... [打印本页]

作者: sunsili    时间: 2021-12-8 10:21
标题: 【RT-Thread】optparse一个开源, 可移植的, 可重入的和可嵌入...
【RT-Thread】optparse一个开源, 可移植的, 可重入的和可嵌入的类getopt命令行参数解析器

optparse是一个开源, 可移植的, 可重入的和可嵌入的类getopt命令行参数解析器. 它支持POSIX getopt选项字符串, GNU风格的长参数解析, 参数置换和子命令处理.


1. 为什么不在RTT下直接使用getopt函数?

POSIX getopt解析器有三个致命的缺陷. 这些缺陷都是可以通过optparse来解决的:

2. 命令顺序

默认情况下, argv在被解析时会将非选项的参数移动到数组的末尾. 可以通过在初始化后将permute字段设置为0来禁用它.

struct optparse options;optparse_init(&options, argv);options.permute = 0;3. 对比

optparse的接口应该是任何熟悉getopt的人都熟悉的. 选项字符串具有相同的格式, 解析器结构体成员变量与getopt全局变量(optarg、optind、optopt)具有相同的名称.

long选项解析器optparse_long()API与GNU的getopt_long()非常相似,可以作为替代api.

optparse没有分配内存. 此外optparse没有依赖项, 包括libc本身. 因此可以在标准C库无法使用的情况下使用.

查看optparse.h文件可以了解完整的API.



4. 示例使用

这是一个通用的getopt命令解析代码:

#include <stdio.h>#include <stdlib.h>#include <stdbool.h>#include <getopt.h>int main(int argc, char **argv){    bool amend = false;    bool brief = false;    const char *color = "white";    int delay = 0;    int option;    while ((option = getopt(argc, argv, "abc:d::")) != -1) {        switch (option) {        case 'a':            amend = true;            break;        case 'b':            brief = true;            break;        case 'c':            color = optarg;            break;        case 'd':            delay = optarg ? atoi(optarg) : 1;            break;        case '?':            return EXIT_FAILURE;        }    }    /* Print remaining arguments. */    for (; optind < argc; optind++)        printf("%s\n", argv[optind]);    return 0;}

如果使用optparse在RTT下解析, 同样的功能将使用如下的代码的实现:

#include <stdio.h>#include <stdlib.h>#include <stdbool.h>#include "optparse.h"int cmd_parse(int argc, char **argv){    bool amend = false;    bool brief = false;    const char *color = "white";    int delay = 0;    char *arg;    int option;    struct optparse options;    optparse_init(&options, argv);    while ((option = optparse(&options, "abc:d::")) != -1) {        switch (option) {        case 'a':            amend = true;            break;        case 'b':            brief = true;            break;        case 'c':            color = options.optarg;            break;        case 'd':            delay = options.optarg ? atoi(options.optarg) : 1;            break;        case '?':            fprintf(stderr, "%s: %s\n", argv[0], options.errmsg);            return EXIT_FAILURE;        }    }    /* Print remaining arguments. */    while ((arg = optparse_arg(&options)))        printf("%s\n", arg);    return 0;}MSH_CMD_EXPORT(cmd_parse, cmd description);

这是optparse对长选项参数的支持:

#include <stdio.h>#include <stdlib.h>#include <stdbool.h>#include "optparse.h"int main(int argc, char **argv){    struct optparse_long longopts[] = {        {"amend", 'a', OPTPARSE_NONE},        {"brief", 'b', OPTPARSE_NONE},        {"color", 'c', OPTPARSE_REQUIRED},        {"delay", 'd', OPTPARSE_OPTIONAL},        {0}    };    bool amend = false;    bool brief = false;    const char *color = "white";    int delay = 0;    char *arg;    int option;    struct optparse options;    optparse_init(&options, argv);    while ((option = optparse_long(&options, longopts, NULL)) != -1) {        switch (option) {        case 'a':            amend = true;            break;        case 'b':            brief = true;            break;        case 'c':            color = options.optarg;            break;        case 'd':            delay = options.optarg ? atoi(options.optarg) : 1;            break;        case '?':            fprintf(stderr, "%s: %s\n", argv[0], options.errmsg);            return EXIT_FAILURE;        }    }    /* Print remaining arguments. */    while ((arg = optparse_arg(&options)))        printf("%s\n", arg);    return 0;}

5. 感谢






欢迎光临 谷动谷力 (http://bbs.sunsili.com/) Powered by Discuz! X3.2