linux调用脚本后如何获得脚本上的输出,ping的使用

2024-08-24 16:32

本文主要是介绍linux调用脚本后如何获得脚本上的输出,ping的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

linux c程序中获取shell脚本输出

1. 前言

  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调用脚本后如何获得脚本上的输出,ping的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1103004

相关文章

Python使用getopt处理命令行参数示例解析(最佳实践)

《Python使用getopt处理命令行参数示例解析(最佳实践)》getopt模块是Python标准库中一个简单但强大的命令行参数处理工具,它特别适合那些需要快速实现基本命令行参数解析的场景,或者需要... 目录为什么需要处理命令行参数?getopt模块基础实际应用示例与其他参数处理方式的比较常见问http

C 语言中enum枚举的定义和使用小结

《C语言中enum枚举的定义和使用小结》在C语言里,enum(枚举)是一种用户自定义的数据类型,它能够让你创建一组具名的整数常量,下面我会从定义、使用、特性等方面详细介绍enum,感兴趣的朋友一起看... 目录1、引言2、基本定义3、定义枚举变量4、自定义枚举常量的值5、枚举与switch语句结合使用6、枚

使用Python从PPT文档中提取图片和图片信息(如坐标、宽度和高度等)

《使用Python从PPT文档中提取图片和图片信息(如坐标、宽度和高度等)》PPT是一种高效的信息展示工具,广泛应用于教育、商务和设计等多个领域,PPT文档中常常包含丰富的图片内容,这些图片不仅提升了... 目录一、引言二、环境与工具三、python 提取PPT背景图片3.1 提取幻灯片背景图片3.2 提取

使用Python实现图像LBP特征提取的操作方法

《使用Python实现图像LBP特征提取的操作方法》LBP特征叫做局部二值模式,常用于纹理特征提取,并在纹理分类中具有较强的区分能力,本文给大家介绍了如何使用Python实现图像LBP特征提取的操作方... 目录一、LBP特征介绍二、LBP特征描述三、一些改进版本的LBP1.圆形LBP算子2.旋转不变的LB

Maven的使用和配置国内源的保姆级教程

《Maven的使用和配置国内源的保姆级教程》Maven是⼀个项目管理工具,基于POM(ProjectObjectModel,项目对象模型)的概念,Maven可以通过一小段描述信息来管理项目的构建,报告... 目录1. 什么是Maven?2.创建⼀个Maven项目3.Maven 核心功能4.使用Maven H

Python中__init__方法使用的深度解析

《Python中__init__方法使用的深度解析》在Python的面向对象编程(OOP)体系中,__init__方法如同建造房屋时的奠基仪式——它定义了对象诞生时的初始状态,下面我们就来深入了解下_... 目录一、__init__的基因图谱二、初始化过程的魔法时刻继承链中的初始化顺序self参数的奥秘默认

SpringBoot使用GZIP压缩反回数据问题

《SpringBoot使用GZIP压缩反回数据问题》:本文主要介绍SpringBoot使用GZIP压缩反回数据问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录SpringBoot使用GZIP压缩反回数据1、初识gzip2、gzip是什么,可以干什么?3、Spr

Linux命令之firewalld的用法

《Linux命令之firewalld的用法》:本文主要介绍Linux命令之firewalld的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux命令之firewalld1、程序包2、启动firewalld3、配置文件4、firewalld规则定义的九大

Spring Boot 集成 Quartz并使用Cron 表达式实现定时任务

《SpringBoot集成Quartz并使用Cron表达式实现定时任务》本篇文章介绍了如何在SpringBoot中集成Quartz进行定时任务调度,并通过Cron表达式控制任务... 目录前言1. 添加 Quartz 依赖2. 创建 Quartz 任务3. 配置 Quartz 任务调度4. 启动 Sprin

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at