cç¨åºè°ç¨shellèæ¬å ±æ两ç§æ¹æ³
ï¼system()ãpopen()ï¼åå«å¦ä¸ï¼
system()
ä¸ç¨èªå·±å»å建è¿ç¨ï¼ç³»ç»å·²ç»å°è£ äºè¿ä¸æ¥ï¼ç´æ¥å å ¥èªå·±çå½ä»¤å³å¯
popen()
ä¹å¯ä»¥å®ç°æ§è¡çå½ä»¤ï¼æ¯system
å¼éå°
以ä¸åå«è¯´æï¼
1ï¼system(shellå½ä»¤æshellèæ¬è·¯å¾ææ¶è¿)
system()
ä¼è°ç¨fork()产ç
ååç¨ï¼ç±ååç¨æ¥è°ç¨/bin/sh-c
stringæ¥å±¥è¡
åæ°stringå符串æ代表çå½ä»¤ï¼æ¤å½ä»¤å±¥è¡
å®åéå³è¿ååè°ç¨çåç¨ãå¨è°ç¨ç©ºæ¢system()æé´sigchld
ä¿¡å·ä¼è¢«ææ¶æç½®ï¼sigintåsigquit
ä¿¡å·åä¼è¢«æ¼ è§
ã
è¿
åå¼ï¼å¦æsystem()å¨è°ç¨/bin/shæ¶å¤±è´¥åè¿å127ï¼å ¶ä»å¤±è´¥åå è¿å-1ãè¥åæ°string为空æé(null)ï¼åè¿åéé¶å¼ã
å¦æ
system()è°ç¨æå
åæåä¼è¿åå±¥è¡
shellå½ä»¤åçè¿åå¼ï¼ä½æ¯æ¤è¿åå¼ä¹æå¯è½ä¸ºsystem()è°ç¨/bin/sh失败æè¿åç127ï¼å
æ¤æ好è½ååç
errno
æ¥ç¡®è®¤å±¥è¡
æå
ã
systemå½ä»¤ä»¥å ¶ç®ç¥
é«æçä½ç¨å¾å°å¾å¾å¹¿æ³
çå©ç¨
ï¼ä¸é¢æ¯ä¸ä¸ªä¾å
ä¾ï¼å¨/tmp/testdir/ç®å½ä¸æshellèæ¬tsh.shï¼å 容为
#!/bin/sh
wget
$1
echo
"done!"
2ï¼popen(char
*command,char
*type)
popen()
ä¼è°ç¨fork()产ç
ååç¨ï¼ç¶åä»ååç¨ä¸è°ç¨/bin/sh
-cæ¥å±¥è¡
åæ°commandçæ令ãåæ°typeå¯åºç¨
ârâ代表读åï¼âwâ代表åå ¥ãéµå¾ªæ¤typeå¼ï¼popen()ä¼å»ºç«
管éè¿å°ååç¨çæ å
è¾åºè®¾å¤
ææ å
è¾å ¥è®¾å¤
ï¼ç¶åè¿åä¸ä¸ªæ件æéãéååç¨ä¾¿å¯å©ç¨
æ¤æ件æéæ¥è¯»åååç¨çè¾åºè®¾å¤
ææ¯åå ¥å°ååç¨çæ å
è¾å ¥è®¾å¤
ä¸ãæ¤å¤ï¼ææåºç¨
æ
件æé(file*)æä½çå½æ°ä¹é½å¯ä»¥åºç¨
ï¼é¤äºfclose()以桥éå¤ã
è¿åå¼ï¼è¥æå
åè¿åæ件æéï¼å¦åè¿ånullï¼å·®é
åå åäºerrnoä¸ã注æï¼å¨ç¼åå ·suid/sgidæéçç¨åºæ¶è¯·å°½éé¿å åºç¨
popen()ï¼popen()ä¼ç»§æ¿ç¯å¢åéï¼éè¿ç¯å¢åéå¯è½ä¼é æç³»ç»å®å ¨çé®é¢ã
ä¾ï¼cç¨åºpopentest.cå 容å¦ä¸ï¼
#include
main
{
file
*
fp
charbuffer[80]
fp=popen(â~/myprogram/test.shâ,ârâ)
fgets(buffer,sizeof(buffer),fp)
printf(â%sâ,buffer)
pclose(fp)
}
在 C 语言坦扒中,第一行不需要写入 shell 脚本程序的#!/bin/bash,是因让锋昌为 C 语言程序是由 C 编译器而不是 shell 来执行的shell脚本一般第一行包含一个指示解释器用来执行基帆脚本的命令,#!/bin/bash表示脚本应该使用 bash 解释器执行
C语言中调用shell指令,根据调用指令目的,可以区分如下两种情况:
一、需要shell指令执行某一功能,如创建文件夹,或者删除文件夹等,程序中不关注shell指令的输出,那么可以使用system函数。
system函数声明于stdlib.h, 功能为调用系统命令,形式为
int system(const char *cmd)
其中cmd为要执行的命令字符串,返回值为执行是否成功的标记。
比如在Linux下要删除当前文件夹下的所有扩展名为a的文件,即*.a, 可以写作
system("rm *.a -f")二、不仅要执行shell命令,还需要得知运行的打印结果,并在程序中使用。
对于此,有两种方案:
1、用system命令,将输出重定向到一个txt文件中,执行后,再读取txt文件,使用后删除。
比如Linux下获取剩余内存的指令可以写作:
system("free>result.txt")//结果重定向到result.txt中。FILE *fp = fopen("result.txt", "r")//打开文件。
int r
while(fgetc(fp) != '\n') //忽略第一行。
fscanf(fp, "%*s%*d%*d%d",&r)//读取第四个域的值,即剩余内存值。
printf("剩余内存为%d KB\n",r)//打印结果。
fclose(fp)//关闭文件。
unlink("result.txt")//删除临时文件。
2、使用重定向,需要经过磁盘读写,还要删除文件,相对低效。同时还有可能出现临时文件和已有文件重名,导致误删数据的情况。 所以一般使用更方便快捷的方式,即调用popen。
FILE *popen(const char *cmd, const char *mode)
使用popen的功能和system类似,属于方法1中执行命令和打开文件的一个组合。不过这里用到的文件是隐式的,并不会在系统中真正存在。返回的指针即结果文件指针。 当使用pclose关闭后,文件自动销毁。
方法1中的例子,用popen实现如下:
FILE *fp = popen("free", "r")//执行命令,同时创建管道文件。int r
while(fgetc(fp) != '\n') //忽略第一行。
fscanf(fp, "%*s%*d%*d%d",&r)//读取第四个域的值,即剩余内存值。
printf("剩余内存为%d KB\n",r)//打印结果。
pclose(fp)//关闭并销毁管道文件。
三、注意事项:
虽然调用shell命令有时可以大大减少代码量,甚至有千行坦则正代码不如一句shell的说法,不过调用shell命令还是有局限性的:
1、使用shell命令会调用盯陆系统资源,效率偏低;
2、不同平台的shell指令不同,导致可移植性下让悔降;
3、调用shell命令时会复制当前进程(fork),如果当前进程的资源占有比较大,会导致瞬间资源占用极大,甚至可能出现失败。
所以,在编码时,除非是测试性的代码,否则在正式代码中不建议使用shell。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)