Linux函数popen pclose学习_pclose函数(1)

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前在阿里

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Linux运维全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上运维知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化的资料的朋友,可以点击这里获取!

一、概念
#include <stdio.h>
FILE * popen ( const char * command , const char * type );
int pclose ( FILE * stream );

popen() 函数通过创建一个管道,调用 fork 产生一个子进程,执行一个 shell 以运行命令来开启一个进程。

  • command 参数是一个指向以 NULL 结束的 shell 命令字符串的指针。这行命令将被传到 bin/sh 并使用-c 标志,shell 将执行这个命令。
  • type 参数只能是读或者写中的一种,得到的返回值(标准 I/O 流)也具有和 type 相应的只读或只写类型。如果 type 是 “r” 则文件指针连接到 command 的标准输出;如果 type 是 “w” 则文件指针连接到 command 的标准输入。
  • popen 的返回值是个标准 I/O 流,必须由 pclose 来终止,否则会产生僵尸子进程。pclose调用只在popen启动的进程结束后才返回。
  • 当使用popen()时,不要屏蔽SIGCHLD信号,popen()使用fork()创建了子进程来运行所给的命令,需要通过此信号判断子进程是否已经退出。

核心:标准I/O函数库提供的popen函数,本质上是对无名管道的使用,它创建一个管道并启动另外一个进程,该进程要么从该管道读出标准输入,要么从该管道写入标准输出。popen在调用进程和指定命令之间创建一个管道。需要主要的使用该方法的缺点:指定命令将出错信息写到标准错误输出,而popen不对标准错误输出做任何处理,它仅仅重定向标准输出。

二、读写示例
1. 读取外部程序的输出

我们在程序中用popen访问uname命令给出的信息。命令uname -a的作用是打印系统信息,包括计算机型号、操作系统名称、版本和发行号、以及计算机网络名。
完成程序的初始化工作后,打开一个连接到uname命令的管道,把管道设置为可读方式并让read_fp指向该命令的输出。最后,关闭read_fp指向的管道。

#include <sys/types.h> 
#include <unistd.h> 
#include <stdlib.h> 
#include <stdio.h> 
#include <string.h> 

int main()    
{    
    FILE   *read_fp;    
    char   buf[1024];   
    int    chars_read;

    memset(buf, '\0', sizeof(buf) );//初始化buf,以免后面写如乱码到文件中 
    read_fp = popen( "uname -a", "r" ); //将“uname -a”命令的输出通过管道读取(“r”参数)到FILE\* stream 
    if(read_fp != NULL)
    {
        chars_read = fread( buf, sizeof(char), sizeof(buf),  read_fp);//将数据流读取到buf中 
        if(chars_read >0)
            printf("my output:\n%s\n",buf);

            pclose( read_fp );    
    }

    return 0;  
}   

这里写图片描述

2. 将输出送往外部程序

这里将数据写入管道,使用的额是od(八进制输出)的命令。

#include <sys/types.h> 
#include <unistd.h> 
#include <stdlib.h> 
#include <stdio.h> 
#include <string.h> 

int main()    
{    
    FILE   *write_fp;    
    char   buf[1024];   
    int    chars_write;

    memset(buf, '\0', sizeof(buf) );//初始化buf,以免后面写如乱码到文件中
    sprintf(buf,"hello world...\n");  
    write_fp = popen( "od -c", "w" ); //通过管道(“w”参数)写入到FILE\* stream 
    if(write_fp != NULL)
    {
        fwrite( buf, sizeof(char), sizeof(buf),  write_fp);  //将FILE\* write\_fp的数据流写入到buf中 
        pclose( write_fp );    
    }

    return 0;  
}   

程序使用带有参数“w”的popen启动od -c命令,这样就可以向该命令发送数据了。然后它给od -c命令发送一个字符串,该命令接收并处理它,最后把处理结果打印到自己的标准输出上。
在命令行上,我们可以使用下面命令得到同样的输出结果:
echo “hello world…\n” | od -c

三、返回值分析

和system调用类似,也需要考虑调用返回值。popen执行一个 shell 以运行命令来开启一个进程。pclose() 函数关闭标准 I/O 流,等待命令执行结束,然后返回 shell 的终止状态。如果 shell 不能被执行,则 pclose() 返回的终止状态与 shell 已执行 exit 一样。

这里重点参考一位博主的文章,首先要明确几点:

  • pclose 失败返回 -1, 成功则返回 exit status, 同 system 类似,需要用 WIFEXITED, WEXITSTATUS 等获取命令返回值。
  • 和 system 一样,SIGCHLD 依然会影响 popen,如果设置了SIGCHLD 则获取不到子进程的状态。
  • 管道只能处理标准输出,不能处理标准错误输出。
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <signal.h>

int main(int argc, char* argv[])
{
    char cmd[1024];
    char line[1024];
    FILE* pipe;
    int rv;

    if (argc != 2)
    {
        printf("Usage: %s <path>\n", argv[0]);
        return -1;
    }

    // pclose fail: No child processes
    //signal(SIGCHLD, SIG\_IGN);

    snprintf(cmd, sizeof(cmd), "ls -l %s 2>/dev/null", argv[1]);

    pipe = popen(cmd, "r");
    if(NULL == pipe)
    {
        printf("popen() failed: %s\n", cmd);
        return -1;
    }

    while(fgets(line, sizeof(line),pipe) != NULL)
    {
        printf("%s", line);
    }



**网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。**

**[需要这份系统化的资料的朋友,可以点击这里获取!](https://bbs.csdn.net/topics/618635766)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

提升。**

**[需要这份系统化的资料的朋友,可以点击这里获取!](https://bbs.csdn.net/topics/618635766)**

**一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!**

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值