如何在C语言中执行shell命令

如何在C语言中执行shell命令,第1张

在c语言中调�”.shell命令的方法实现。

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。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/12368943.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-24
下一篇 2023-05-24

发表评论

登录后才能评论

评论列表(0条)

保存