http://doc.chinaunix.net/linux/201004/512284.shtml1. 前言

  Unix界有一句名言:“一行shell脚本胜过万行C程序”,虽然这句话有些夸张,但不可否认的是,借助脚本确实能够极大的简化一些编程工作。比如实现一个ping程序来测试网络的连通性,实现ping函数需要写上200~300行代码,为什么不能直接调用系统的ping命令呢?通常在程序中通过 system函数来调用shell命令。但是,system函数仅返回命令是否执行成功,而我们可能需要获得shell命令在控制台上输出的结果。例如,执行外部命令ping后,如果执行失败,我们希望得到ping的返回信息。

  2. 使用临时文件

  首先想到的方法就是将命令输出重定向到一个临时文件,在我们的应用程序中读取这个临时文件,获得外部命令执行结果,代码如下所示:

  #define CMD_STR_LEN 1024

  int mysystem(char* cmdstring, char* tmpfile)

  {

  char cmd_string[CMD_STR_LEN];

  tmpnam(tmpfile);

  sprintf(cmd_string, "%s > %s", cmdstring, tmpfile);

  return system(cmd_string);

  }

  这种使用使用了临时文件作为应用程序和外部命令之间的联系桥梁,在应用程序中需要读取文件,然后再删除该临时文件,比较繁琐,优点是实现简单,容易理解。有没有不借助临时文件的方法呢?

  3. 使用匿名管道

  在<<UNIX环境高级编程>>一书中给出了一种通过匿名管道方式将程序结果输出到分页程序的例子,因此想到,我们也可以通过管道来将外部命令的结果同应用程序连接起来。方法就是fork一个子进程,并创建一个匿名管道,在子进程中执行shell命令,并将其标准输出dup 到匿名管道的输入端,父进程从管道中读取,即可获得shell命令的输出,代码如下:

  /**   * 增强的system函数,能够返回system调用的输出   *

  * @param[in] cmdstring 调用外部程序或脚本的命令串

  * @param[out] buf 返回外部命令的结果的缓冲区

  * @param[in] len 缓冲区buf的长度

  *   * @return 0: 成功; -1: 失败    */

  int mysystem(char* cmdstring, char* buf, int len)

  {

  int   fd[2]; pid_t pid;

  int   n, count;

  memset(buf, 0, len);

  if (pipe(fd) < 0)

  return -1;

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

  return -1;

  else if (pid > 0)     /* parent process */

  {

  close(fd[1]);     /* close write end */

  count = 0;

  while ((n = read(fd[0], buf + count, len)) > 0 && count > len)

  count += n;

  close(fd[0]);

  if (waitpid(pid, NULL, 0) > 0)

  return -1;

  }

  else    /* child process */

  {

  close(fd[0]);     /* close read end */

  if (fd[1] != STDOUT_FILENO)

  {

  if (dup2(fd[1], STDOUT_FILENO) != STDOUT_FILENO)

  {

  return -1;

  }

  close(fd[1]);

  }

  if (execl("/bin/sh", "sh", "-c", cmdstring, (char*)0) == -1)

  return -1;

  }

  return 0;

  }

  4. 使用popen

  在学习unix编程的过程中,发现系统还提供了一个popen函数,可以非常简单的处理调用shell,其函数原型如下:

  FILE *popen(const char *command, const char *type);

  该函数的作用是创建一个管道,fork一个进程,然后执行shell,而shell的输出可以采用读取文件的方式获得。采用这种方法,既避免了创建临时文件,又不受输出字符数的限制,推荐使用。

  popen使用FIFO管道执行外部程序。

  #include <stdio.h>

  FILE *popen(const char *command, const char *type);

  int pclose(FILE *stream);

  popen 通过type是r还是w确定command的输入/输出方向,r和w是相对command的管道而言的。r表示command从管道中读入,w表示 command通过管道输出到它的stdout,popen返回FIFO管道的文件流指针。pclose则用于使用结束后关闭这个指针。

  下面看一个例子:

  #include <sys/types.h>

  #include <unistd.h>

  #include <stdlib.h>

  #include <stdio.h>

  #include <string.h>

  int main( void )

  {

  FILE   *stream;

  FILE    *wstream;

  char   buf[1024];

  memset( buf, '\0', sizeof(buf) );//初始化buf,以免后面写如乱码到文件中

  stream = popen( "ls -l", "r" ); //将“ls -l”命令的输出 通过管道读取(“r”参数)到FILE* stream

  wstream = fopen( "test_popen.txt", "w+"); //新建一个可写的文件

  fread( buf, sizeof(char), sizeof(buf), stream); //将刚刚FILE* stream的数据流读取到buf中

  fwrite( buf, 1, sizeof(buf), wstream );//将buf中的数据写到FILE    *wstream对应的流中,也是写到文件中

  pclose( stream );

  fclose( wstream );

  return 0;

  }

  [root@localhost src]# gcc popen.c

  [root@localhost src]# ./a.out

  [root@localhost src]# cat test_popen.txt

  总计 128

  -rwxr-xr-x 1 root root 5558 09-30 11:51 a.out

  -rwxr-xr-x 1 root root 542 09-30 00:00 child_fork.c

  -rwxr-xr-x 1 root root 480 09-30 00:13 execve.c

  -rwxr-xr-x 1 root root 1811 09-29 21:33 fork.c

  -rwxr-xr-x 1 root root 162 09-29 18:54 getpid.c

  -rwxr-xr-x 1 root root 1105 09-30 11:49 popen.c

  -rwxr-xr-x 1 root root 443 09-30 00:55 system.c

  -rwxr-xr-x 1 root root    0 09-30 11:51 test_popen.txt

  -rwxr-xr-x 1 root root 4094 09-30 11:39 test.txt

  5. 小结

  有统计数据表明,代码的缺陷率是一定的,与所使用的语言无关。Linux提供了很多的实用工具和脚本,在程序中调用工具和脚本,无疑可以简化程序,从而降低代码的缺陷数目。Linux shell脚本也是一个强大的工具,我们可以根据需要编制脚本,然后在程序中调用自定义脚本。

linux c程序中获取shell脚本输出相关推荐

  1. inux C程序中获取shell脚本输出(如获取system命令输出)

    1. 前言 Unix 界有一句名言:"一行shell脚本胜过万行C程序",虽然这句话有些夸张,但不可否认的是,借助脚本确实能够极大的简化一些编程工作.比如实现一个 ping程序来测 ...

  2. c中获取python控制台输出_linux c程序中获取shell脚本输出的实现方法

    [转]Linux Shell脚本调试技术 本文转载自:https://www.ibm.com/developerworks/cn/linux/l-cn-shell-debug/ Shell脚本调试技术 ...

  3. linux shell 获取表,bash - 如何获取shell脚本中目录中的文件列表?

    bash - 如何获取shell脚本中目录中的文件列表? 我试图使用shell脚本获取目录的内容. 我的脚本是: for entry in `ls $search_dir`; do echo $ent ...

  4. 如何获取shell脚本中某条语句的执行时间

    有没有办法获取shell脚本中某条语句的执行时间 如果要获得语句执行所花的时间 用time命令 在语句的前面加上time命令 输出结果一共3列 比如我执行time sleep 5 real    0m ...

  5. Linux 中启用 Shell 脚本的调试模式

    shell 脚本调试系列 Linux 中启用 Shell 脚本的调试模式 在 Shell 脚本中执行语法检查调试模式 在 Shell 脚本中跟踪调试命令的执行 概述 脚本是存储在一个文件的一系列命令. ...

  6. Linux中使用shell脚本生成随机数

    Linux中使用shell脚本生成随机数 1.主体思想: 为了实现随机数的随机性,所以在这里借用date中的%s%N可以输出从1970年1月1日00:00:00到目前经历的秒数,从而产生变化的数字,再 ...

  7. Shell中获取当前脚本的绝对路径: $( cd “$( dirname “${BASH_SOURCE[0]}“ )“ pwd)

    Shell中获取当前脚本的绝对路径:  $( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd) 参考链 ...

  8. Linux shell脚本输出日志的方法和案例

    Linux shell脚本输出日志 #### log_correct函数打印正确的输出到日志文件 function log_correct () {DATE=`date "+%Y-%m-%d ...

  9. Linux中执行shell脚本的4种方法

    这篇文章主要介绍了Linux中执行shell脚本的4种方法总结,即在Linux中运行shell脚本的4种方法,需要的朋友可以参考下. bash shell 脚本的方法有多种,现在作个小结.假设我们编写 ...

最新文章

  1. oracle用户登录的认证方式
  2. 理解“动心忍性”的含义
  3. rtsp摘要认证协议(Response计算方法)
  4. thinkphp数据库连接池_Thinkphp5框架下的数据库连接
  5. (17)FPGA面试技能提升篇(System Verilog)
  6. 浙江理工大学机械复试c语言真题,2016年浙江理工大学机械与自动控制学院计算机应用基础)之C程序设计复试笔试最后押题五套卷...
  7. 2022 github新建账号技巧
  8. 算法:翻转图片Rotate Image
  9. SpringCloud OpenFeign(二)
  10. 服务器系统怎么安装网卡驱动,网卡驱动怎么安装,教您网卡驱动的安装操作
  11. JavaScript的toast
  12. 淘宝评论爬取 python pandas
  13. 【论文笔记】MV3D:Multi-View 3D Object Detection Network for Autonomous Driving
  14. FreeSWITCH安装教程
  15. 华为路由器:ISIS基本原理与配置(含实验)
  16. 51单片机——电子密码锁
  17. 热力学统计物理专题:热力学统计物理(I)知识结构
  18. 网站布局工具的对比:Flexbox VS Susy
  19. 免费好用的短视频去水印工具
  20. 智能电动牙刷方案功能特点

热门文章

  1. python苹果下载软件助手哪个好_Mac上有什么实用的必备软件?
  2. Linux常用命令:FireWall
  3. 删除所有数据_批量删除空白单元格,只会Ctrl+G定位就out了,全部三种方法都在这...
  4. 计算机网络-思维导图(3)数据链路层
  5. 2022 年是 Linux 桌面年吗?
  6. 实战|Python轻松实现动态网页爬虫(附详细源码)
  7. 一张表看尽CV和NLP的经典+前沿论文,还教你阅读顶会论文,构建深度学习知识框架...
  8. 阿里算法工程师公开机器学习路线,你的路走对了吗?
  9. 下列不是python对文件的写操作方法的是_Python—文件读写操作
  10. 交大网院计算机第五次作业答案,交大网院计算机第三次作业分析.docx