前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >L001 Linux和android ndk 外部程序调用popen 和system的用法

L001 Linux和android ndk 外部程序调用popen 和system的用法

作者头像
上善若水.夏
发布2018-09-28 10:52:11
2.7K0
发布2018-09-28 10:52:11
举报
文章被收录于专栏:上善若水上善若水

我们支持,linux常用的调用外部指令,有两个接口ppen和system

popen的使用

函数原型

代码语言:javascript
复制
#include <stdio.h>
FILE * popen ( const char * command , const char * type );
int pclose ( FILE * stream );

说明

popen() 函数通过创建一个管道,调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程。这个进程必须由 pclose() 函数关闭,而不是 fclose() 函数。pclose() 函数关闭标准 I/O 流,等待命令执行结束,然后返回 shell 的终止状态。如果 shell 不能被执行,则 pclose() 返回的终止状态与 shell 已执行 exit 一样。

type 参数只能是读或者写中的一种,得到的返回值(标准 I/O 流)也具有和 type 相应的只读或只写类型。如果 type 是 "r" 则文件指针连接到 command 的标准输出;如果 type 是 "w" 则文件指针连接到 command 的标准输入。

command 参数是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用-c 标志,shell 将执行这个命令。

popen 的返回值是个标准 I/O 流,必须由 pclose 来终止。前面提到这个流是单向的。所以向这个流写内容相当于写入该命令的标准输入;命令的标准输出和调用 popen 的进程相同。与之相反的,从流中读数据相当于读取命令的标准输出;命令的标准输入和调用 popen 的进程相同。

返回值

如果调用 fork() 或 pipe() 失败,或者不能分配内存将返回NULL,否则返回标准 I/O 流。 返回错误 popen 没有为内存分配失败设置 errno 值。 如果调用 fork() 或 pipe() 时出现错误,errno 被设为相应的错误类型。 如果 type 参数不合法,errno将返回EINVAL。

使用举例

代码语言:javascript
复制
/**运行命令
 *1. cmd    要执行的命令
 *2. 参数type可使用“r”代表读取,“w”代表写入。
 */
int runShell(const char *cmd,const char *type,char *resultBuffer,int len)
{   
    int ret;    
    
    FILE * pp=popen(cmd,type);          
    ret = fread ( resultBuffer, 1, len, pp );  
    pclose(pp);  

    return ret;
}

如果想看到标准出错的信息呢?

只需在命令中加入重定向 例如:

代码语言:javascript
复制
cp -f 2>&1

system调用

system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命>令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。 返回值

代码语言:javascript
复制
=-1:出现错误
=0:调用成功但是没有出现子进程
>0:成功退出的子进程的id 

命令执行的返回值获取可参考我的如下例程:

代码语言:javascript
复制
int mysystem(char *cmd)
{
    pid_t status;  
  
    status = system(cmd);  
    if (-1 == status)  
    {  
        D("system error! status=%d\n",status);  
    }  
    else  
    {  
        
        if (WIFEXITED(status))  
        {  
            if (0 == WEXITSTATUS(status))  
            {  
                D("run shell script successfully.\n");  
                return 0;
            }  
            else  
            {  
                E("run script fail, script exit code: %d\n", WEXITSTATUS(status));  
                return WEXITSTATUS(status);
            }  
        }  
        else  
        {  
            D("exit  %d\n", WEXITSTATUS(status));  
            return WEXITSTATUS(status);
        }  
    }  
}

system调用总是返回-1的原因

返回-1的原因是因为进程将SIGCHLD屏蔽了!

确实我的程序在main函数里面就调用signal(SIGCHLD, SIG_IGN);将SIGCHLD屏蔽了! system的调用过程主要是调三个函数

  1. fork()
  2. execl()
  3. wait()

wait需要依赖SIGCHLD信号,所以屏蔽了SIGCHLD信号就永远得不到system正确的返回值了!

参考链接

  1. Linux的system()和popen()差异
  2. system调用总是返回-1
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2016.05.27 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • popen的使用
    • 函数原型
      • 说明
        • 返回值
          • 使用举例
            • 如果想看到标准出错的信息呢?
            • system调用
              • system调用总是返回-1的原因
              • 参考链接
              领券
              问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档