getopt_long的使用

linux疑难问题排查实战,分享了作为公司专家,在项目开发过程中内存优化(堆、栈、代码段、数据段)、性能优化、死机(栈越界、堆越界)、死锁等疑难问题排查的案例、使用的工具(perf、asan、strace、memleak等)、工作经验,大家可以点开看一下,如果觉得有用可以关注一下,认真学习相信可以让你在日后工作中大放光彩。

头文件:#include <getopt.h>
函数原型:

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

参数解析:

  1. argc、argv直接从main函数中获取。
  2. opting是选项参数组成的字符串,由下列元素组成:
    (1)单个字符,表示选项,
    (2)单个字符后接一个冒号:表示该选项后必须跟一个参数。参数紧跟在选项后或者以空格隔开。该参数的指针赋给optarg。
    (3)单个字符后跟两个冒号,表示该选项后可以有参数也可以没有参数。如果有参数,参数必须紧跟在选项后不能以空格隔开。该参数的指针赋给optarg。(这个特性是GNU的扩张)。
  3. struct option结构体
struct option {
const char *name; //name表示的是长参数名
int has_arg;
//has_arg有3个值,no_argument(或者是0),表示该参数后面不跟参数值
// required_argument(或者是1),表示该参数后面一定要跟个参数值
// optional_argument(或者是2),表示该参数后面可以跟,也可以不跟参数值
int *flag;    
//用来决定,getopt_long()的返回值到底是什么。如果这个指针为NULL,那么getopt_long()返回该结构val字段中的数值。如果该指针不为NULL,getopt_long()会使得它所指向的变量中填入val字段中的数值,并且getopt_long()返回0。如果flag不是NULL,但未发现长选项,那么它所指向的变量的数值不变。
int val;    
//和flag联合决定返回值 这个值是发现了长选项时的返回值,或者flag不是 NULL时载入*flag中的值。典型情况下,若flag不是NULL,那么val是个真/假值,譬如1 或0;另一方面,如 果flag是NULL,那么val通常是字符常量,若长选项与短选项一致,那么该字符常量应该与optstring中出现的这个选项的参数相同。

返回值:

  1. 如getopt_long 返回-1,表示argv[]中的所有选项被解析出。
  2. 如果成功找到了选项,getopt()会返回选项字符。
  3. 如果遇到一个缺少参数的选项字符,返回值就要取决于optstring的第一个字符了:如果是’:’,则返回’:’,否则返回’?’。

内部定义的一些全局变量:
extern char* optarg;
extern int optind, opterr, optopt;
参数optarg:指向当前选项参数的指针
参数optind:再次调用getopt时的下一个argv指针索引
参数optopt:表示最后一个未知选项

例子:

#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>
#include <linux/fb.h>
#include <getopt.h>
#include <signal.h>
#include <errno.h>
#include "client.h"
#include "server.h"
int FRAME_PER_PACKET = 0;
static struct option long_options[] = {
{ "server", 0, 0, 's'},
{ "client_self", 0, 0, 'c'},
{ "client_send", 0, 0, 'd'},
{ "client_recv", 0, 0, 'v'},
{ "PER_FREAM_PACKET", 1, 0, 'p'},
{ "auto test from 1 to 10", 0, 0, 'A'},
{0, 0, 0, 0},
};
static char optstring[] = "scdvp:A";

#define print_opt_help(opt_index, help_str)             \
do {                                \
    printf("\t---%s\t\t-%c\t%s", long_options[opt_index].name, (char)long_options[opt_index].val, help_str); \
} while (0)

void usage() {
    printf("\nUsage:\n");
print_opt_help(0, "action server\n");
print_opt_help(1, "action client self send and self recv\n");
print_opt_help(2, "action client send\n");
print_opt_help(3, "action client recv\n");
print_opt_help(4, "set PER_FREAM_PACKET num\n");
print_opt_help(5, "auto test from 1 to 10\n");

printf("\n\nExamples:\n");
printf("\t./net-test -s\t---\taction server\n");
printf("\t./net-test -c -p 10\t---\taction client PER_FREAM_PACKET is 10\n");
}
int main(int argc, char *argv[]){
    int c = 0;
    client_act client_act_t;
    while(1){
        c = getopt_long(argc, argv, optstring, long_options, NULL);
        switch (c){
            case 's':
                printf("start server\n");
                server_fun();
                break;
            case 'c':
                printf("start client self send and recv test\n");
                                client_act_t = RECV_AND_SEND;
                client_fun(client_act_t);
                break;
            case 'd':
                printf("start client send test\n");
                client_act_t = SEND_ONLY;
                client_fun(client_act_t);
                break;
        case 'v':
                printf("start client recv test\n");
                client_act_t = RECV_ONLY;
                client_fun(client_act_t);
                break;
        case 'A':
                printf("start client auto test\n");
                client_fun(client_act_t);
                break;
        case 'p':
                FRAME_PER_PACKET = atoi(optarg);
                printf("set PER_FREAM_PACKET is %d\n", FRAME_PER_PACKET);
                break;
        default :
                usage();
                return -1;
        }
    }
    return 0;
}

也可参考RTMP源码:

  • 1
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: `getopt_long` 是一个 C 语言标准库中的函数,用于解析命令行参数。与 `getopt` 不同的是,`getopt_long` 可以处理长选项(long options),即以两个减号开头的选项。`getopt_long` 的函数原型为: ```c #include <getopt.h> int getopt_long(int argc, char * const argv[], const char *optstring, const struct option *longopts, int *longindex); ``` 其中,`argc` 和 `argv` 分别是命令行参数的个数和数组,`optstring` 是短选项字符串,`longopts` 和 `longindex` 分别是长选项结构体数组和当前处理的长选项的索引。`getopt_long` 函数会依次解析命令行参数,返回当前处理的选项的字符代码,如果已经处理完所有选项,则返回 -1。如果遇到无效选项或缺少参数,会返回 '?',并在 `optopt` 变量中保存无效选项的字符代码或缺少参数的选项的字符代码。`getopt_long` 函数还会修改 `optind` 变量,指示下一个要处理的命令行参数的索引。 ### 回答2: getopt_long是一个函数库,用于处理命令行参数的解析和处理。它是标准C库中getopt函数的扩展版本,提供更加灵活和强大的功能。 getopt_long可以解析命令行参数的各种形式,包括短选项、长选项和参数。它使用了一个结构体数组来定义所有支持的选项和参数,每个选项和参数都有一个相关的处理函数。这个结构体数组的定义可以通过编码方式来指定,也可以通过外部文件来定义。这使得getopt_long可以灵活适应不同的命令行参数需求。 调用getopt_long函数时,它会依次读取命令行参数,解析出对应的选项和参数,并调用相应的处理函数来进行处理。在解析过程中,它可以处理多个选项和参数的组合,并且支持选项和参数的排列顺序任意。 getopt_long还提供了其他辅助函数来帮助处理命令行参数,比如打印错误信息、获取非选项参数等。 总的来说,getopt_long是一个强大的函数库,可以方便地处理各种形式的命令行参数,并提供了多种功能来满足不同的应用需求。它在编程中的使用可以帮助程序员更加高效地处理命令行参数,提高代码的可维护性和可扩展性。 ### 回答3: getopt_long是一个在C语言中用来解析命令行参数的函数。它是标准库中getopt函数的一个扩展,可以处理更复杂和更灵活的命令行选项。 getopt_long函数包含在头文件<getopt.h>中,使用时需要传入一些参数。其中,长选项是以结构体数组的形式传入函数的,每个结构体包含了选项名、选项标志以及选项值等信息。通过判断输入的命令行参数是否匹配传入的长选项,函数可以进行相应的处理。 getopt_long函数返回一个整型值,当解析完所有选项时返回-1,否则循环解析每个选项,并返回选项的字符值,或者根据参数optstring中的":"和"?"进行错误处理。 getopt_long函数可以处理带有参数的长选项,例如"--output=file",我们可以通过optarg变量获取选项参数的值。另外,它还可以处理短选项,如"-o",并且可以支持多个选项合并在一起,如"-ov"代表同时设置-o和-v两个选项。 通过使用getopt_long函数,我们可以方便而灵活地解析命令行参数,实现不同的功能。它可以在命令行中设置各种选项,例如调试模式、输出文件名、日志级别等等。同时,getopt_long使用方法也相对简单,只需要了解相关参数和函数的用法,就可以轻松地处理命令行参数
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值