getopt函数理解

179 阅读3分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第1天

前瞻知识

main函数 int main (int argc,char *argv[]); argc :代表命令行中输入项的个数(包括可执行文件名) argv: 数组的长度为argc

函数头文件

#include<unistd.h>

函数声明

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

参数理解

前两个是main函数的参数,argv中有“-”开头的参数为选项参数,“ .”后跟随的字符称为选项字符,第三个参数是字符串,一个包含了合法选项字符的字符串. 相关参数 extern char* optarg;//保存选项的参数 extern int optind;//记录下一个检索的位置 extern int opterr;//是否将错误信息输出stderr,非0就表示要把错误输出到stderr上 extern int optopt;//表示不在选项字符串optstring中的选项

选项的理解

例1:linux下对c文件的编译命令是 gcc hello.c -o hello.out -o就是命令行的选项,后面的hello.out就是它所携带的参数. 例2: linux下删除一个文件的命令是 rm 目录 -rf 这两个(r,f)选项都是不带参数的,可以写在一起

optstring参数理解

冒号表示参数,一个冒号表示必须带参数,两个冒号表示参数是可选的(可以有也可以没有)没有冒号就代表不带参数 abc: 就是ab选项是不带参数的,c选项的参数是可选的 a:b:cd::就是ab选项是必须带参数的,c选项不带参数,d选项的参数是可选的

典型用法例子
#include <unistd.h>
#include <stdio.h>
int main(int argc, char * argv[])
{
    
    int ch;
    printf("\n\n");
    printf("optind:%d,opterr:%d\n",optind,opterr);
    printf("--------------------------\n");
       while ((ch = getopt(argc, argv, "ab:c:de::")) != -1)
       //ab:c:de::是opstring参数
       {
        printf("optind: %d\n", optind);
           switch (ch) 
        {
               case 'a':
                       printf("HAVE option: -a\n\n");   
                       break;
               case 'b':
                       printf("HAVE option: -b\n"); 
                       printf("The argument of -b is %s\n\n", optarg);
                       break;
               case 'c':
                       printf("HAVE option: -c\n");
                       printf("The argument of -c is %s\n\n", optarg);
                       break;
               case 'd':
                   printf("HAVE option: -d\n");
                     break;
              case 'e':
                    printf("HAVE option: -e\n");
                    printf("The argument of -e is %s\n\n", optarg);
                  break;
              case '?':
                       printf("Unknown option: %c\n",(char)optopt);
                       break;
               }
       }


}

文件名为main.c,先执行命令gcc main.c -o main编译。 执行./main -b “hello world” 结果为 optind:1,opterr:1---他们两个的初值都为1,表示下一个检索的位置是argv[1],y要讲错误输出到stderr -------------------------: optind: 3 HAVE option: -b The argument of -b is hello world

getopt()会改变argv[]中参数的顺序。经过多次getopt()后,argv[]中的选项和选项的参数会被放置在数组前面,而optind 会指向第一个非选项和参数的位置

#include <unistd.h>
#include <stdio.h>
int main(int argc, char * argv[])
{
    
    int i;
    printf("--------------------------\n");
    for(i=0;i<argc;i++)
    {
        printf("%s\n",argv[i]);
    }
    printf("--------------------------\n");
       //int aflag=0, bflag=0, cflag=0;
    
       int ch;
    printf("\n\n");
    printf("optind:%d,opterr:%d\n",optind,opterr);
    printf("--------------------------\n");
       while ((ch = getopt(argc, argv, "ab:c:de::")) != -1)
       {
        printf("optind: %d\n", optind);
           switch (ch) 
        {
               case 'a':
                       printf("HAVE option: -a\n\n");   
                       break;
               case 'b':
                       printf("HAVE option: -b\n"); 
                       printf("The argument of -b is %s\n\n", optarg);
                       break;
               case 'c':
                       printf("HAVE option: -c\n");
                       printf("The argument of -c is %s\n\n", optarg);
                       break;
               case 'd':
                   printf("HAVE option: -d\n");
                     break;
              case 'e':
                    printf("HAVE option: -e\n");
                    printf("The argument of -e is %s\n\n", optarg);
                  break;
              case '?':
                       printf("Unknown option: %c\n",(char)optopt);
                       break;
               }
       }
    
       printf("----------------------------\n");
      printf("optind=%d,argv[%d]=%s\n",optind,optind,argv[optind]);

    printf("--------------------------\n");
    for(i=0;i<argc;i++)
    {
        printf("%s\n",argv[i]);
    }
    printf("--------------------------\n");
    

}

运行 ./getpot zheng -b "qing er" han -c123 qing 结果如下:

--------------------------
./getpot
zheng
-b
qing er
han
-c123
qing
--------------------------


optind:1,opterr:1
--------------------------
optind: 4
HAVE option: -b
The argument of -b is qing er

optind: 6
HAVE option: -c
The argument of -c is 123

----------------------------
optind=4,argv[4]=zheng
--------------------------
./getpot
-b
qing er
-c123
zheng
han
qing
--------------------------

argv[]中参数的顺序已经改变了 改变的规律是 选项和对应的参数放在了前面,又不是选项又不是参数的按照顺序放在了后面