博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
从Linux程序中执行shell(程序、脚本)并获得输出结果
阅读量:2441 次
发布时间:2019-05-10

本文共 3582 字,大约阅读时间需要 11 分钟。

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命令的输出,代码如下:

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)           {          close(fd[1]);               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                        {          close(fd[0]);               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脚本也是一个强大的工具,我们可以根据需要编制脚本,然后在程序中调用自定义脚本。

转载地址:http://nenqb.baihongyu.com/

你可能感兴趣的文章
本地应用程序_本地化PHP应用程序的“正确方法”,第4部分
查看>>
创业基础网络课答案_创业挑战网络基础设施巨头
查看>>
php使用正则表达式_使用PHP正则表达式
查看>>
用PHP上传文件
查看>>
mfc单文档绘制世界坐标系_文档使世界运转
查看>>
防止跨站请求伪造(CSRF)
查看>>
科创板 云计算业务公司_云计算对您的业务意味着什么
查看>>
SitePoint播客#80:与澳大利亚团队的FullCodePress
查看>>
可以给视图加复合主键_灵活视图处理的模式,第1部分–使用复合材料
查看>>
SitePoint播客#83:与Adam Broadway重新思考您的工作
查看>>
SitePoint播客#117:与Paul Boag返回Boagworld
查看>>
支付宝沙箱设置_从零到云:设置EC2沙箱,第1部分
查看>>
bcrypt强哈希_为什么应该使用Bcrypt哈希存储的密码
查看>>
数据库存储集合_处理集合根的集合-存储库模式
查看>>
SitePoint播客#76:邪恶的WordPress主题
查看>>
BigCommerce和Volusion:比较设置和移动
查看>>
PHP开发人员的8个分布式计算谬论
查看>>
SitePoint播客#123:ASP.NET与Tim Posey和Pranav Rastogi
查看>>
使用codeigniter_使用CodeIgniter探索面向方面的编程,第3部分
查看>>
使用codeigniter_使用CodeIgniter探索面向方面的编程,第2部分
查看>>