为什么以下shell命令在命令行中直接执行时为何有效,而在使用popen / system通过C程序执行时却无效?

时间:2019-09-27 18:28:48

标签: c linux command-line adb popen

命令是:ps -c -p | tr -s " " | cut -d " " -f 2,6-10,13 | grep 'R'

我正在通过adb shell运行它。基本上,我想要一个当前在运行队列中的进程(和某些参数)的列表。如果我直接通过外壳运行它,则工作正常。

但是,如果我将其放在C程序中并交叉编译以在Android上运行,它将无法正常工作。只有ps -c -p在起作用(我已经检查过了)。但是运行此ps -c -p | tr -s " " | cut -d " " -f 2,6-10,13 | grep 'R'时,我得到了输出:

usage: tr [-cds] SET1 [SET2]

Translate, squeeze, or delete characters from stdin, writing to stdout

-c/-C  Take complement of SET1
-d     Delete input characters coded SET1
-s     Squeeze multiple output characters of SET2 into one character

tr: Needs 1 argument
usage: cut OPTION... [FILE]...

Print selected parts of lines from each FILE to standard output.

-b LIST select only these bytes from LIST.
-c LIST select only these characters from LIST.
-f LIST select only these fields.
-d DELIM    use DELIM instead of TAB for field delimiter.
-s  do not print lines not containing delimiters.
-n  don't split multibyte characters (Ignored).

cut: Needs -fcb

我认为ps -c -p的输出未传送到tr,后者也未传送到cut。 你能建议出什么问题吗?

这是我正在使用的代码:

    #include <stdio.h>
    #include <time.h>
    #include <stdlib.h>
    #include <string.h>

    #define BUFSIZE 128

    int main(int argc,char **argv)
    {
        char *cmd4 = "ps -c -p | tr -s " " | cut -d " " -f 2,6-10,13 | grep 'R'";    
        system(cmd4);

        FILE *fp;


        char buf[BUFSIZE];
     // Another method
    if ((fp = popen(cmd4, "r")) == NULL) {
        printf("Error opening pipe4!\n");
        return -1;
    }

    while (fgets(buf, BUFSIZE, fp) != NULL) {
        // Do whatever you want here...
        printf("cmd 4 running!");
        printf("OUTPUT: %s", buf);
    }

    if(pclose(fp))  {
        printf("Command not found or exited with error status4\n");
        return -1;
    }

    return 0;
}

1 个答案:

答案 0 :(得分:2)

在外壳中,您正在使用以下命令:

ps -c -p | tr -s " " | cut -d " " -f 2,6-10,13 | grep 'R'

在C语言中,您将以下内容传递给system(后来传递给popen):

ps -c -p | tr -s  | cut -d  -f 2,6-10,13 | grep 'R'

看到区别了吗?引号需要在C源代码中转义。此外,当您遇到此类麻烦时,请确保输出相关数据,以便可以看到实际发生的情况而不是计划的情况。一个简单的puts(cmd4)就可以立即发现这一点。