C C++最新Linux函数popen pclose学习_pclose函数,2024年最新京东面试真题解析

img
img

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

需要这份系统化的资料的朋友,可以添加戳这里获取

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

#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


![](https://img-blog.csdn.net/20180204132939304?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdTAxMzQ1NzE2Nw==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast)


##### **三、返回值分析**


和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);
}

rv = pclose(pipe);

if (-1 == rv)
{
    printf("pclose() failed: %s\n", strerror(errno));
    return -1;
}

if (WIFEXITED(rv))
{
    printf("subprocess exited, exit code: %d\n", WEXITSTATUS(rv));
    if (0 == WEXITSTATUS(rv))
    {
        // if command returning 0 means succeed
        printf("command succeed\n");
    }
    else
    {
        if(127 == WEXITSTATUS(rv))
        {
            printf("command not found\n");
            return WEXITSTATUS(rv);
        }
        else
        {
            printf("command failed: %s\n", strerror(WEXITSTATUS(rv)));
            return WEXITSTATUS(rv);
        }
    }
}
else
{
    printf("subprocess exit failed\n");
    return -1;
}

return 0;

}


##### **四、总结**


总的来说,请求popen调用运行一个程序时,它首先启动shell,即系统的shell命令,然后将command字符串作为一个参数传递给它。这样就有了优缺点:   
 **优点**是:由于所有类Unix系统中参数扩展都是由shell完成的,所有它运行我们通过popen完成非常复杂的shell命令。而其他一些创建进程的函数(如execl)调用起来就复杂的多,因为调用进程必须自己完成shell扩展。   
 **缺点**是:针对每个popen调用,不仅要启动一个被请求的程序,还要启动一个shell,即每个popen调用将启动两个进程。从节省系统资源的角度来看,popen函数的调用成本略高,并且对目标命令的调用比正常方式慢一些(通过pipe改进)。


和system相比,system就是执行shell命令最后返回是否执行成功,popen执行命令并且通过管道和shell命令进行通信。








![img](https://img-blog.csdnimg.cn/img_convert/c0d915e58dbd257172a6acc89ce09d71.png)
![img](https://img-blog.csdnimg.cn/img_convert/cef67e30e4b63a4a5bd5309194cac02b.png)

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

**[需要这份系统化的资料的朋友,可以添加戳这里获取](https://bbs.csdn.net/topics/618668825)**


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

以添加戳这里获取](https://bbs.csdn.net/topics/618668825)**


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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值