GNU getopt 不填充 optarg
GNU getopt not populating optarg
我正在尝试使用 GNU getopt
函数在我的 C 程序中解析一些命令行参数。假设我拨打了以下电话:
./my_program -E 16 -t path/to/file
我目前的期望是,如果我有以下代码:
long arg;
int e_val;
char *t_str;
while ((arg = getopt(argc, argv, "E:t:")) != -1) {
switch(arg) {
case 'E':
e_val = *optarg; // e_val = 16
break;
case 't':
t_str = optarg;
break;
}
}
那么 e_val
将等于 16
而 t_str
将指向 path/to/file
。
但是,我似乎无法弄清楚如何访问这些参数的实际值。当我开始在 case 'E'
中分配 e_val
时,optarg
仍然是空指针:
Starting program: /stack/overflow/example -E 16 -t Whosebug
Breakpoint 1, get_arguments (argc=5, argv=0x7fffffffe318) at stack_overflow.c:233
233 while ((arg = getopt(argc, argv, "E:t:")) != -1)
(gdb) n
235 switch (arg)
(gdb)
254 e_val = *optarg;
(gdb) p optarg
= 0x0
(gdb) x/1xw optarg
0x0: Cannot access memory at address 0x0
(gdb) p /x arg
= 0x45 // ASCII for 'E'
为什么没有设置 optarg?如果 optarg
不是我应该用来获得 E
之后的 16
的值,那么我应该使用什么?
作为最后的说明,我用 gdb 简单地遍历了 getopt
,看起来它出于某种原因没有设置 extern optarg
变量。
回溯:
#0 _getopt_internal (argc=<value optimized out>, argv=<value optimized out>, optstring=<value optimized out>,
longopts=<value optimized out>, longind=<value optimized out>, long_only=<value optimized out>, posixly_correct=0)
at getopt.c:1135
#1 0x00007ffff7894138 in getopt (argc=<value optimized out>, argv=<value optimized out>, optstring=<value optimized out>)
at getopt.c:1145
#2 0x0000000000400cd6 in get_arguments (argc=5, argv=0x7fffffffe318) at Whosebug.c:233
#3 0x0000000000400875 in main (argc=5, argv=0x7fffffffe318) at Whosebug.c:82
正在执行的代码:
(gdb) l
1130
1131 result = _getopt_internal_r (argc, argv, optstring, longopts,
1132 longind, long_only, &getopt_data,
1133 posixly_correct);
1134
1135 optind = getopt_data.optind;
1136 optarg = getopt_data.optarg;
1137 optopt = getopt_data.optopt;
1138
1139 return result;
optarg
未设置:
(gdb) n
1136 optarg = getopt_data.optarg;
(gdb)
1135 optind = getopt_data.optind;
(gdb) p optarg
= 0x0
(gdb) p getopt_data.optarg
= 0x7fffffffe60d "16"
请注意,此处未设置 optarg
并非由于编译器优化。我可以一路回到我的函数,它保持为空。
这不是一个真正的答案,但它太大而不能作为评论。
此代码在严格的编译选项下编译干净:
#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc, char **argv)
{
long arg;
int e_val = -1;
char *t_str = "pygmalion";
while ((arg = getopt(argc, argv, "E:t:")) != -1)
{
switch (arg)
{
case 'E':
e_val = atoi(optarg); // e_val = 16
break;
case 't':
t_str = optarg;
break;
}
}
printf("e_val = %d\n", e_val);
printf("t_str = %s\n", t_str);
return 0;
}
编译并运行:
$ gcc -g -O3 -std=c11 -Wall -Wextra -Wmissing-prototypes -Wstrict-prototypes \
> -Wold-style-definition -Wold-style-declaration -Werror go.c -o go
$ ./go -E 16 -t path/to/file
e_val = 16
t_str = path/to/file
$ ./go -E 16
e_val = 16
t_str = pygmalion
$ ./go -t pygmalion
e_val = -1
t_str = pygmalion
$ ./go
e_val = -1
t_str = pygmalion
$
使用 GCC 5.1.0 在 Ubuntu 14.04 LTS 衍生产品上进行测试。因此,从表面上看,代码大致是正确的。这反过来表明问题出在周围的代码中。您正在链接的东西是否也定义了 optarg
?
如果您使用我刚刚展示的代码并 运行 它会得到什么?如果它可以工作但你的更大的程序没有,那么你的更大的程序中有一些东西以某种方式引起了麻烦。
我正在尝试使用 GNU getopt
函数在我的 C 程序中解析一些命令行参数。假设我拨打了以下电话:
./my_program -E 16 -t path/to/file
我目前的期望是,如果我有以下代码:
long arg;
int e_val;
char *t_str;
while ((arg = getopt(argc, argv, "E:t:")) != -1) {
switch(arg) {
case 'E':
e_val = *optarg; // e_val = 16
break;
case 't':
t_str = optarg;
break;
}
}
那么 e_val
将等于 16
而 t_str
将指向 path/to/file
。
但是,我似乎无法弄清楚如何访问这些参数的实际值。当我开始在 case 'E'
中分配 e_val
时,optarg
仍然是空指针:
Starting program: /stack/overflow/example -E 16 -t Whosebug
Breakpoint 1, get_arguments (argc=5, argv=0x7fffffffe318) at stack_overflow.c:233
233 while ((arg = getopt(argc, argv, "E:t:")) != -1)
(gdb) n
235 switch (arg)
(gdb)
254 e_val = *optarg;
(gdb) p optarg
= 0x0
(gdb) x/1xw optarg
0x0: Cannot access memory at address 0x0
(gdb) p /x arg
= 0x45 // ASCII for 'E'
为什么没有设置 optarg?如果 optarg
不是我应该用来获得 E
之后的 16
的值,那么我应该使用什么?
作为最后的说明,我用 gdb 简单地遍历了 getopt
,看起来它出于某种原因没有设置 extern optarg
变量。
回溯:
#0 _getopt_internal (argc=<value optimized out>, argv=<value optimized out>, optstring=<value optimized out>,
longopts=<value optimized out>, longind=<value optimized out>, long_only=<value optimized out>, posixly_correct=0)
at getopt.c:1135
#1 0x00007ffff7894138 in getopt (argc=<value optimized out>, argv=<value optimized out>, optstring=<value optimized out>)
at getopt.c:1145
#2 0x0000000000400cd6 in get_arguments (argc=5, argv=0x7fffffffe318) at Whosebug.c:233
#3 0x0000000000400875 in main (argc=5, argv=0x7fffffffe318) at Whosebug.c:82
正在执行的代码:
(gdb) l
1130
1131 result = _getopt_internal_r (argc, argv, optstring, longopts,
1132 longind, long_only, &getopt_data,
1133 posixly_correct);
1134
1135 optind = getopt_data.optind;
1136 optarg = getopt_data.optarg;
1137 optopt = getopt_data.optopt;
1138
1139 return result;
optarg
未设置:
(gdb) n
1136 optarg = getopt_data.optarg;
(gdb)
1135 optind = getopt_data.optind;
(gdb) p optarg
= 0x0
(gdb) p getopt_data.optarg
= 0x7fffffffe60d "16"
请注意,此处未设置 optarg
并非由于编译器优化。我可以一路回到我的函数,它保持为空。
这不是一个真正的答案,但它太大而不能作为评论。
此代码在严格的编译选项下编译干净:
#include <getopt.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(int argc, char **argv)
{
long arg;
int e_val = -1;
char *t_str = "pygmalion";
while ((arg = getopt(argc, argv, "E:t:")) != -1)
{
switch (arg)
{
case 'E':
e_val = atoi(optarg); // e_val = 16
break;
case 't':
t_str = optarg;
break;
}
}
printf("e_val = %d\n", e_val);
printf("t_str = %s\n", t_str);
return 0;
}
编译并运行:
$ gcc -g -O3 -std=c11 -Wall -Wextra -Wmissing-prototypes -Wstrict-prototypes \
> -Wold-style-definition -Wold-style-declaration -Werror go.c -o go
$ ./go -E 16 -t path/to/file
e_val = 16
t_str = path/to/file
$ ./go -E 16
e_val = 16
t_str = pygmalion
$ ./go -t pygmalion
e_val = -1
t_str = pygmalion
$ ./go
e_val = -1
t_str = pygmalion
$
使用 GCC 5.1.0 在 Ubuntu 14.04 LTS 衍生产品上进行测试。因此,从表面上看,代码大致是正确的。这反过来表明问题出在周围的代码中。您正在链接的东西是否也定义了 optarg
?
如果您使用我刚刚展示的代码并 运行 它会得到什么?如果它可以工作但你的更大的程序没有,那么你的更大的程序中有一些东西以某种方式引起了麻烦。