讨厌"的Shell"可爱"的C----用C偷取Shell的结果输
2012-05-28 10:45
513 查看
我自己很喜欢用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用于使用结束后关闭这个指针.实例代码如下所示:
3.使用popen
思想:<<UNIX环境高级编程>>一书中给出了一种通过匿名管道方式将程序结果输出到分页程序的例子,因此想到,我们也可以通过管道来将外部命令的结果同应用程序连接起来。方法就是fork一个子进程,并创建一个匿名管道,在子进程中执行shell命令,并将其标准输出dup到匿名管道的输入端,父进程从管道中读取,即可获得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
相关文章推荐
- "指针+int " 的结果详解。
- Shell组件的"模式"样式
- "report is being generated " 但就是不显示结果的原因
- shell中的"2>&1"是什么意思?
- "自信比能力更重要,知识比财富更重要,诚实比勤奋更重要,宽容比原谅更重要,过程比结果更重要"
- shell中的"2>&1"是什么意思?
- 练手毛坯作品基于LINUX的"QQ闹眼子版本"(包括服务器与客户端<图形版与SHELL文字版>)
- Shell组件的"模式"样式
- 关于Shell 脚本中的"[: too many arguments"错误
- UNIX环境高级编程学习之第四章文件和目录-用C实现Shell中的"ls -l"功能
- "if"决定结果
- shell "cmd 2>&1 >file" "cmd > file 2>&1" 的区别
- Lucene+paoding 使用"庖丁解牛" 构建Analyzer paoding
- "没事,我来扛"
- "clang: error: linker command failed with exit code 1 (use -v to see invocation)"
- 使用truss、strace或ltrace诊断软件的"疑难杂症"
- A标签启动FTP下载 <a href=" ftp://帐号:密码@FTP网址“
- "正版破解"软件下载站大全
- "网卡eth0,未识别的端口"的解决方法
- "undefined reference to strptime"之自己定义strptime函数