3

如何让我的程序接受带空格的命令行参数?

另一个编辑:我刚刚认识到该程序是从一个 shell 脚本启动的,该脚本为程序的执行设置了环境。由于有一些外部库,LD_LIBRARY_PATH设置为当前工作目录:

#!/bin/sh

ARCH=`uname -m`
export LD_LIBRARY_PATH=".:lib/magic/linux-${ARCH}"

./data_sniffer.bin $*

exit $?

这个问题肯定与$*. 是否有任何解决方案可以正确转发命令行参数?

这是来自的代码片段main()

if (stat(argv[1], &lStat) != 0)
{   
    fprintf(stderr, "Cannot stat(2) given file: %s. \n", argv[1]);
    return EXIT_FAILURE;
}   

我正在使用以下参数启动我的程序:

./data_sniffer /mnt/pod/movies/some\ test\ movie\ file.avi

生成的错误消息如下所示:Cannot stat(2) given file: /mnt/pod/movies/some.

有人知道这里有什么问题吗?我认为我不是第一个遇到这个问题的人(虽然,我在这里找不到相关的问题)。

4

2 回答 2

8

在包装脚本中替换使用./data_sniffer.bin $*with./data_sniffer.bin "$@"并且参数应该以正确的方式转发。

更多关于两者的区别$*,可以在这里找到。$@"$@"


这个问题的作者完全改变了它,并提出了有关此事的更多信息,我会让所有已经写好的东西都保持不变,但请记住这是在他/她最后一次编辑之前写的。


关于argv..

作为开发人员,它对你的要求并不高,我很想(而且更真实)说它不需要

传递给应用程序的参数由执行二进制文件的 shell 处理,在下面执行此操作肯定会按照您希望的方式工作(尽管我觉得您声称当前的 shell 不能正确处理很奇怪'\ '):

./data_sniffer '/mnt/pod/movies/some test movie file.avi'
./data_sniffer /mnt/pod/movies/some\ test\ movie\ file.avi

# there should be no difference between the two

我的建议

  • 您是否尝试过printf ("argv[1]: %s\n", argv[1]);在 main 的开头做来验证它的内容?

  • 您确定使用正确的命令行参数调用正确的二进制文件吗?

可悲的是,写的唯一合理的事情是你做错了什么。但是,如果没有有关该问题的进一步信息,我们将无法回答。

我发现很难相信你的shell中有一个错误,即使这当然是可能的——我对此表示怀疑。


将命令行解析为argv开发人员不必担心,您不必为二进制文件本身实现强制功能来处理其参数中的空格。

于 2012-07-14T12:54:18.630 回答
4

在我的 CentOS 5.3 中,我做了一个测试。

#include <stdio.h>
int main(int argc, char** argv)
{
        printf("argc=%d\n", argc);
        int i = 0;
        for(i = 0; i<argc; i++)
        {   
                printf("argv[%d]=%s\n", i, argv[i]);
        }   
        return 0;
}

然后使用不同的参数运行它:

[root@dw examples]# ./a.out a b c
argc=4
argv[0]=./a.out
argv[1]=a
argv[2]=b
argv[3]=c
[root@dw examples]# ./a.out 'a b c'
argc=2
argv[0]=./a.out
argv[1]=a b c
[root@dw examples]# ./a.out a\ b\ c
argc=2
argv[0]=./a.out
argv[1]=a b c
[root@dw examples]# ./a.out "a b c"
argc=2
argv[0]=./a.out
argv[1]=a b c

所以看起来像 (1) 'ab c' ; (2) a\b\c;(3) "ab c" 都在工作。

于 2012-07-14T13:35:19.013 回答