webbench压力测试工具源码剖析(二)--对命令行函数get_opt,字符串处理函数strstr等分析

本文深入探讨了命令行参数解析函数get_opt和getopt_long的使用方法,以及字符串处理函数strstr、strchr、strcspn的功能和应用场景。同时,介绍了i/o处理中的setvbuf函数和fdopen函数的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

       第一篇是对源码逻辑和结构的梳理,这次主要是对里面重要的函数进行分析。比如说对命令行进行处理的getopt_long函数。在整个工具里面,很多地方只是对格式的处理。在获取命令行,分割命令,然后把命令或者参数有填充到请求报文段中。

  • 命令行处理函数get_opt,getopt_long。
  • 字符串处理函数strstr,strchr,strcspn。
  • i/o处理时的setvbuf函数。
  • fdopen()的作用,与fopen()的区别。

1.命令行处理函数get_opt,getopt_long

get_opt函数:函数功能:用来解析命令行参数,参数argc和argv分别代表参数个数和内容,跟main()函数里的命令行参数一样

函数所在头文件:#include<unistd.h>

函数原型定义:   int getopt(int argc, char* const argv[ ], const char *optstring )

参数optstring: 为选项字符串,告知getopt可以处理那个选项以及哪个选项需要参数。

如果选项字符串里的字母后接着冒号:“:”,则表示还有相关的参数全域变量optarg即会指向此额外参数如果在处理期间遇到了不符合optstring指定的其他选项,getopt()将会显示一个错误消息,并将全局域变量设置为“?”字符,将全局域变量opterr设置为0则将不会打印出错信息。

extern char* optarg;

extern int optind, opterr, optopt;

参数optarg:指向当前选项参数的指针。

参数optind:再次调用getopt时的下一个argv指针索引

参数optopt:表示最后一个未知选项。

参数optstring: 比如getopt(argc, argv, "td:ch:q::"),这个主要就是表示你的命令行参数都会有什么参数。下面例子:

1. 单个字符,表示选项,这里一共有t、d、c、h、q五个选项

2. 单个字符后接一个冒号“:”表示该选项后必须跟一个参数,参数紧跟在选项后或者以空格隔开

3. 单个字符后跟两个冒号,表示该选项后可以跟一个参数,也可以不跟,如果后边跟一个参数,参数必须紧跟在选项后不能以空格隔开

#include <stdio.h>
#include<unistd.h>
 
int main(int argc, char *argv[])
{
    int ch;
    opterr = 0;//不打印错误信息
    
    while((ch = getopt(argc,argv,"tl:cm:q::"))!= -1)
    {
        switch(ch)
        {
            //第一个参数t后面不跟参数
            case 't': printf("love IT\n");break;
            
            //第二个参数l后面要跟参数执行命令并打印参数
            case 'l': printf("love life: %s\n",optarg); break;
            
            case 'c': printf("love Code\n"); break;
            case 'm': printf("love myself %s\n", optarg); break;
            
            //第五个参数q后面可跟可不跟参数,一个参数的话必须紧跟q后面不能有空格 
            case 'q': printf("love quite: %s\n",optarg); break;
            
            default: printf("other option :%c\n",ch);
        }
    
    }
    return 0;
}

getopt_long函数
函数功能:支持长选项的命令解析

函数所在头文件:#include<getopt.h>

函数原型定义:int getopt_long(int argc, char * const argv[],const char *optstring, const struct option *longopts,int *longindex)

参数optstring: 为选项字符串,告知getopt_long可以处理那个选项以及哪个选项需要参数,如果选项字符串里的字母后接着冒号:“:”,则表示还有相关的参数,全域变量optarg即会指向此额外参数,如果在处理期间遇到了不符合optstring指定的其他选项,getopt_long()将会显示一个错误消息,并将全局域变量设置为“?”字符,将全局域变量opterr设置为0则将不会打印出错信息。

参数longopts:

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()的返回值到底是什么。如果flag是null(通常情况),则函数会返回与该项option匹配的 val值;如果flag不是NULL,则将val值赋予flag所指向的内存,并且返回值设置为0。
int val; //和flag联合决定返回值
}

举个例子:
struct option long_options[] = {
{"a123", required_argument, 0, 'a'},
{"c123", no_argument, 0, 'c'},
};
现在,如果命令行的参数是-a 123,那么调用getopt_long()将返回字符'a',并且将字符串123由optarg返回(注意注意!字符串123由optarg带回!optarg不需要定义,在getopt.h中已经有定义),那么,如果命令行参数是-c,那么调用getopt_long()将返回字符'c',而此时,optarg是null。最后,当getopt_long()将命令行所有参数全部解析完成后,返回-1。

required_argument(或者是1)时,参数输入格式为:--参数 值 或者 --参数=值。
optional_argument(或者是2)时,参数输入格式只能为:--参数=值。

参数longindex: 表示当前长参数在longopts中的索引值

extern char* optarg;

extern int optind, opterr, optopt;

参数optarg:指向当前选项参数的指针

参数optind:再次调用getopt_long时的下一个argv指针索引

参数optopt:表示最后一个未知选项

参数optstring: 同上


原文链接:https://blog.youkuaiyun.com/coding__madman/article/details/51043733

以上为转载,下面为补充!

2.strstr,strchr,strcspn字符串处理函数。

strstr函数包含着头文件<string.h>中

原型是:char *strstr(const char*haystack,const char *needle);

作用:用于判断needle是否为haystack的子串。

返回值: 如果是子串,则返回一个指针指向子串在haystack中第一次出现的开始的位置。否则NULL被返回。

strchr函数:<string.h>

原型:char *strchr(const char*s,int c);

作用:找到c第一次出现在s中的位置并且返回。

返回值:如果找到就返回第一次出现的位置,否则返回NULL。如果c本身是‘\0’,那么直接返回指向s尾部的指针。既*s=‘\0’。

strspn函数:<string.h>
strspn() 函数用来计算字符串 str 中连续有几个字符都属于字符串 accept,

其原型为:size_t strspn(const char *str, const char * accept);

说明strspn() 从参数 str 字符串的开头计算连续的字符,而这些字符都完全是 accept 所指字符串中的字符。简单的说,若 strspn() 返回的数值为n,则代表字符串 str 开头连续有 n 个字符都是属于字符串 accept 内的字符。

返回值:返回字符串 str 开头连续包含字符串 accept 内的字符数目。所以,如果 str 所包含的字符都属于 accept,那么返回 str 的长度;如果 str 的第一个字符不属于 accept,那么返回 0。区分大小写。

strcspn函数:<string.h>

原型:size_t strcspn(const char *sourc,const char *reject);

作用:与strspn相反,在字符串sourc中起始部分连续不在reject字符串的长度。该函数可以用来查看某个字符在指定字符串中第一次出现的位置。

返回值:返回的是不连续在reject的长度。

3.setvbuf函数。

原型:int setvbuf(FILE *stream,char *buf,int mode,size_t size);

作用:设置file 的i/o的缓冲格式,每次缓冲的大小,缓冲到的空间。

       缓冲区可以设置为无缓冲区,或者行缓冲区或者满缓冲区。对应的三种模式是:_ONBF,_IOLBF,_IOFBF。
       满缓冲区是指当往这个的文件中写值或者读值时,如果设置了缓冲区,会先给设定的缓冲区中写入。当设定的缓冲区满后,会一次性将数据发送到文件中。
       当是行缓冲区时,写完一行一发送。默认的i/o函数一般都是行缓冲区。
       当无缓冲区时,也就是当有数据是立即接收,不用保存在缓冲区中。
通过这样的设置可以改变读取i/o的次数。

4.fdopen函数

原型:FILE *fdopen(int fd,const char *mode);

作用:   将文件描述符转换为一个文件指针,也就是流指针。主要的作用是可以通过文件指针提供一种行缓存模式。在http,sip,ftp的协议中,都是文本传输协议。协议的发送信息很多基本都有换行,表示一个信息分割。那么此时用write/read函数发送的时候它不是行模式发送。发送过去后对方可能还需要检查信息中的换行,然后分别信息。此时可以将fd转化为file*,从而可以用这个指针调用一些行缓存的函数,比如fgets()获得一个字符串知道出现\n或者是末尾。fputs()发送一个字符串。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值