getopt_long支持长选项的命令行解析使用man getopt_long得到其声明如下

#include <getopt.h>

int getopt_long(int argc, char * const argv[],

const char *optstring,

const struct option *longopts, int *longindex);

int getopt_long_only(int argc, char * const argv[],

const char *optstring,

const struct option *longopts, int *longindex);

说明函数中的argc和argv通常直接从main()到两个参数传递而来。optsting是选项参数组成的字符串如

果该字符串里任一字母后有冒号那么这个选项就要求有参数。下一个参数是指向数组的指针这个数组是

option结构数组option结构称为长选项表其声明如下

struct option {

const char *name;

int has_arg;

int *flag;

int val;

};

结构中的元素解释如下

const char *name选项名前面没有短横线。譬如"help"、"verbose"之类。

int has_arg描述长选项是否有选项参数如果有是哪种类型的参数其值见下表:

符号常量 数值 含义

no_argument 0 选项没有参数

required_argument 1 选项需要参数

optional_argument 2 选项参数是可选的

int *flag

如果该指针为NULL那么getopt_long返回val字段的值

如果该指针不为NULL那么会使得它所指向的结构填入val字段的值同时getopt_long返回0

int val

如果flag是NULL那么val通常是个字符常量如果短选项和长选项一致那么该字符就应该与optstring中

出现的这个选项的参数相同

最后一个参数longindex参数一般赋为NULL即可如果没有设置为NULL那么它就指向一个变量这个变量

会被赋值为寻找到的长选项在longopts中的索引值这可以用于错误诊断。

注GNU提供的getopt-long()和getopt-long-only()函数其中后者的长选项字串是以一个短横线开始的

而非一对短横线。

linux 命令行约定

几乎所有的GNU/Linux程序都遵循一些命令行参数定义的约定。程序希望出现的参数可以分成两种选

项options or flags、其他类型的的参数。Options修饰了程序运行的方式其他类型的参数则提供了输

入例如输入文件的名称。

对于options类型参数可以有两种方式

1短选项short options:顾名思义就是短小参数。它们通常包含一个连字号和一个字母大写

或小写字母。例如-s-h等。

2长选项long options长选项包含了两个连字号和一些大小写字母组成的单词。例如--

size--help等。

*注一个程序通常会提供包括short options和long options两种参数形式的参数。

对于其他类型参数的说明

这种类型的参数通常跟随在options类型参数之后。例如ls –s /功能为显示root目录的大小。’/

’这个参数告诉ls要显示目录的路径。

getopt_long()函数使用规则

1使用前准备两种数据结构

字符指针型变量

该数据结构包括了所有要定义的短选项每一个选项都只用单个字母表示。如果该选项需要参数如

需要文件路径等则其后跟一个冒号。例如三个短选项分别为‘-h’‘-o’‘-v’其中-o需要参数

其他两个不需要参数。那么我们可以将数据结构定义成如下形式

const char * const shor_options = “ho:v” ;

struct option 类型数组

该数据结构中的每个元素对应了一个长选项并且每个元素是由四个域组成。通常情况下可以按以下

规则使用。第一个元素描述长选项的名称第二个选项代表该选项是否需要跟着参数需要参数则为1

反之为0第三个选项可以赋为NULL第四个选项是该长选项对应的短选项名称。另外数据结构的最后

一个元素要求所有域的内容均为0即{NULL,0,NULL,0}。下面举例说明还是按照短选项为‘-h’‘-o’

‘-v’的例子该数据结构可以定义成如下形式

const struct option long_options = {

{ “help”, 0, NULL, ‘h’ },

{ “output”, 1, NULL, ‘o’ },

{ “verbose”, 0, NULL, ‘v’ },

{ NULL, 0, NULL, 0 }

};

2调用方法

参照1准备的两个数据结构则调用方式可为

getopt_long( argc, argv, short_options, long_options, NULL);

3几种常见返回值

(a)每次调用该函数它都会分析一个选项并且返回它的短选项如果分析完毕即已经没有选项了

则会返回-1。

(b)如果getopt_long()在分析选项时遇到一个没有定义过的选项则返回值为‘?’此时程序员可

以打印出所定义命令行的使用信息给用户。

(c)当处理一个带参数的选项时全局变量optarg会指向它的参数

(d)当函数分析完所有参数时全局变量optindinto argv会指向第一个‘非选项’的位置

实践小例子

view plaincopy to clipboardprint?

#include <stdio.h>

#include <getopt.h>

char *l_opt_arg;

char* const short_options = "nbl:";

struct option long_options[] = {

{ "name", 0, NULL, 'n' },

{ "bf_name", 0, NULL, 'b' },

{ "love", 1, NULL, 'l' },

{ 0, 0, 0, 0},

};

int main(int argc, char *argv[])

{

int c;

while((c = getopt_long (argc, argv, short_options, long_options, NULL)) != -1)

{

switch (c)

{

case 'n':

printf("My name is XL.\n");

break;

case 'b':

printf("His name is ST.\n");

break;

case 'l':

l_opt_arg = optarg;

printf("Our love is %s!\n", l_opt_arg);

break;

}

}

return 0;

}

#include <stdio.h>

#include <getopt.h>

char *l_opt_arg;

char* const short_options = "nbl:";

struct option long_options[] = {

{ "name", 0, NULL, 'n' },

{ "bf_name", 0, NULL, 'b' },

{ "love", 1, NULL, 'l' },

{ 0, 0, 0, 0},

};

int main(int argc, char *argv[])

{

int c;

while((c = getopt_long (argc, argv, short_options, long_options, NULL)) != -1)

{

switch (c)

{

case 'n':

printf("My name is XL.\n");

break;

case 'b':

printf("His name is ST.\n");

break;

case 'l':

l_opt_arg = optarg;

printf("Our love is %s!\n", l_opt_arg);

break;

}

}

return 0;

}

编译并运行

[root@localhost liuxltest]# gcc -o getopt getopt.c

[root@localhost liuxltest]# ./getopt -n -b -l forever

My name is XL.

His name is ST.

Our love is forever!

[root@localhost liuxltest]#

[root@localhost liuxltest]# ./getopt -nb -l forever

My name is XL.

His name is ST.

Our love is forever!

[root@localhost liuxltest]# ./getopt -nbl forever

My name is XL.

His name is ST.

Our love is forever!

?'ag?Y?[yle='font-size:10.0pt; font-family:宋体;mso-hansi-font-family:"Times New Roman";mso-bidi-font-family: 宋体'> {

return name;

}

int get_score() const

{

return score;

}

private:

string name;

int score;

};

// output student's name and score

void output_student( const Student& student )

{

cout << student.get_name() << "\t";

cout << student.get_score() << endl;

}

int main()

{

Student stu( "Wang", 85 );

output_student( stu );

}

[/code:1:edc740afb5]

设计了一个类 Student数据成员有 name 和 score有两个构造函数有一个设置成员数据函数 set_student()各有一个取得 name 和 score 的函数 get_name() 和 get_score()。请注意 get_name() 和 get_score() 后面都加了 const而 set_student() 后面没有也不能有const。

首先说一点题外话为什么 get_name() 前面也加 const。如果没有前后两个 const 的话get_name() 返回的是对私有数据成员 name 的引用所以通过这个引用可以改变私有成员 name 的值如

[code:1:edc740afb5] Student stu( "Wang", 85 );

stu.get_name() = "Li";

[/code:1:edc740afb5]

即把 name 由原来的 "Wang" 变成了 "Li"而这不是我们希望的发生的。所以在 get_name() 前面加 const 避免这种情况的发生。

那么get_name() 和 get_score() 这两个后面应该加 const的成员函数如果没有 const 修饰的话可不可以呢回答是可以但是这样做的代价是const对象将不能再调用这两个非const成员函数了。如

[code:1:edc740afb5]const string& get_name(); // 这两个函数都应该设成 const 型

int get_score();

void output_student( const Student& student )

{

cout << student.get_name() << "\t"; // 如果 get_name() 和 get_score() 是非const成员函数这一句和下一句调用是错误的

cout << student.get_score() << endl;

}

[/code:1:edc740afb5]

由于参数student表示的是一个对const Student型对象的引用所以 student 不能调用非const成员函数如 set_student()。如果 get_name() 和 get_score() 成员函数也变成非const型那么上面的 student.get_name() 和 student.get_score() 的使用就是非法的这样就会给我们处理问题造成困难。

因此我们没有理由反对使用const该加const时就应该加上const这样使成员函数除了非const的对象之外const对象也能够调用它。

转自