第一篇是对源码逻辑和结构的梳理,这次主要是对里面重要的函数进行分析。比如说对命令行进行处理的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()发送一个字符串。