Linux 下的system和popen

本文介绍了Linux中的system和popen函数,它们都用于执行shell命令,但工作方式有所不同。system函数通过fork和execl执行命令,而popen创建管道与shell命令进行I/O交互。system函数执行后立即返回,popen则返回一个文件指针,允许连续读写。此外,文章还提供了这两个函数的使用示例。
摘要由CSDN通过智能技术生成

一、Linux system函数

system函数可以简单理解为利用system执行shell 命令
int system(const char *command);

1.源码简单分析

来看下面linux版system函数的源码:

int system(const char * cmdstring)
{
  pid_t pid;
  int status;

  if(cmdstring == NULL){
      
      return (1);
  }


  if((pid = fork())<0){

        status = -1;
  }
  else if(pid == 0){
    execl("/bin/sh", "sh", "-c", cmdstring, (char *)0);
    -exit(127); //子进程正常执行则不会执行此语句
    }
  else{
        while(waitpid(pid, &status, 0) < 0){
          if(errno != EINTER){
            status = -1;
            break;
          }
        }
    }
    return status;
}

由linux源码可见:
当system接受的命令为NULL时直接返回,否则fork出一个子进程,父进程使用waitpid等待子进程结束,子进程则是调用execl来启动一个程序代替自己,execl("/bin/sh", “sh”, “-c”, cmdstring,(char*)0)是调用shell,这个shell的路径是/bin/sh,后面的字符串都是参数,然后子进程就变成了一个 shell进程,这个shell的参数是cmdstring,就是system接受的参数。
execl函数具体介绍点击

2.简单说明

函数说明
system()会调用fork()产生子进程,由子进程来调用execl函数来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。

返回值
如果system()在调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果 system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。

system和execl函数目的都是执行shell命令,但很显然system函数更加简单粗暴。但二者还是有点不同的。比如:调用system执行完后随即返回原调用的进程,而execl执行完毕则不会返回。

二、Linux popen函数

函数原型
FILE popen( const char * command, const char* mode )

参数说明:

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

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

返回值

如果调用成功,则返回一个读或者打开文件的指针,如果失败,返回NULL,具体错误要根据errno判断。
调用成功后用pclose关闭文件IO流-----pclose (FILE* stream)

作用
popen函数作用与system函数作用都是执行一个shell命令。但popen() 函数用于创建一个管道:其内部实现为调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程这个进程必须由 pclose() 函数关闭。
也就是popen创建了一个管道,将执行命令后的数据流流入这个管道中,通过标准C库函数fread可以读取数据。同时popen执行完后,仍然可以向管道继续传送命令,实现功能。

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

int main()
{
        FILE *fp;
        char readBuf[1024];
        memset(readBuf,'\0',strlen(readBuf));

        fp = popen("ls -l","r");
        if(fp == NULL){
                return 0;
        }
        // 读取管道中的数据
        fread(readBuf,1,1024,fp);
        printf("read:\n%s\n",readBuf);
        pclose(fp);
        return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值