讨厌"的Shell"可爱"的C----用C偷取Shell的结果输

       我自己很喜欢用C语言,也羡慕小王老是说的那个shell这强大,那强大.一直不服他,凭啥这么说,有C语言强大么,C语言都能些操作系统系统,你Shell能吗?每次这个时候,小王就傻笑着对我摇摇头说: 说不过你,看你上次写过什么网卡什么的ping程序,这不,我电脑不能上网了,你用C写个程序ping一下子,我看看。我说那好办(心里想,我上次才写的博客,才分析的linux内核网络协议,这不是小case),包我身上了。接下来的将近一个小时,小王在背后不时地先笑笑,后皱皱眉,做回撇撇嘴不屑的样子,就这么重复着像是死循环跟我杠上了,好歹功夫不负有心人,出来了,结构是她电脑上网正常。这结果,给我气的是那个叫天叫地的,我容易么,小王?

       只见小王还在傻笑,不慌不忙的说:你好费经啊,这简单的问题,看你用C语言弄的,看我的:ping XXX.XXX.XXX.XXX.咦!咋这简单,一下子就出来了,结果比我的还好理解多了,高傲的我..沮丧啊,伤心啊,没面子啊。算了, 转念一想,我能不能在我的C程序里调用这个Shell命令或这调用这个命令的结果呢?说干就干,记得以前看谭浩强那本C,写老师布置的菜单程序时,需要用到系统的clear命令,当时是system(“clear”); 我这里试试?嗳,果然可以,但是返回来的只有调用命令执行的失败和成功,没有具体结果,这可不好在小王面前耍,继续找,最后还真让我找到了下面几种方法:

1.使用临时文件

思想 : 在命令行中,不是有个重定向操作>>么,那我就将命令的结果重定向到一个临时文件tmp中,然后用C语言的文件接口去读就好了,读完了之后,再删       除这个命令不是挺好。这种方法简单实用,但用到了第三个临时文件,总觉得不完美。
2.使用匿名管道

思想:在linux系统中,还提供了一个系统调用popen(),可以非常简单的处理调用shell,其函数原型如下: 
FILE *popen(const char *command, const char *type); 
该函数的作用是创建一个管道,fork一个进程,然后执行shell,而shell的输出可以采用读取文件的方式获得。采用这种方法,既避免了创建临时文件,又不受输出字符数的限制,推荐使用。popen使用FIFO管道执行外部程序,通过type是r还是w确定command的输入输出方向,其中type是相对于command的管道而言的。r表示command从管道中读入,对应w也是。函数返回FIFO管道的文件流指针。对应的pclose用于使用结束后关闭这个指针.实例代码如下所示:  

#省去头文件
int main( void )
{
  FILE   *stream;
  FILE   *wstream;
  char   buf[1024];
  memset( buf, '\0', sizeof(buf) );                      //初始化buf,以免后面写如乱码到文件中
  stream = popen( "ls -l", "r" );                        //将“ls -l”命令的输出通过管道读取到stream
  wstream = fopen( "test_popen.txt", "w+");   //新建一个可写的文件
  fread( buf, sizeof(char), sizeof(buf), stream);  //将刚刚FILE* stream的数据流读取到buf中
  fwrite( buf, 1, sizeof(buf), wstream );             //将buf中的数据写到流wstream中,也是写到文件中
  pclose( stream );
  fclose( wstream );
  return 0;
}

3.使用popen

思想:<<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]);     /* 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;
}
       有了上面三种方法,我又可以在小王面前炫耀一次啦,嘿嘿,我不仅用了shell,而且用到了我习惯的C,最后,这篇博客参考了如下文章,大家知道就可以了,千万别告诉小王,想想,其实我挺不容易的,程序的我们围着可爱的表妹,其中的酸甜,你我都知,最后给出参考文章地址: http://www.embeddedlinux.org.cn/html/jishuzixun/201008/31-855.html

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值