文档章节

【C/C++】Linux下使用system()函数一定要谨慎

恋恋美食
 恋恋美食
发布于 2012/04/15 00:35
字数 1346
阅读 90556
收藏 34
曾经的曾经,被system()函数折磨过,之所以这样,是因为对system()函数了解不够深入。只是简单的知道用这个函数执行一个系统命令,这远远不够,它的返回值、它所执行命令的返回值以及命令执行失败原因如何定位,这才是重点。当初因为这个函数风险较多,故抛弃不用,改用其他的方法。这里先不说我用了什么方法,这里必须要搞懂system()函数,因为还是有很多人用了system()函数,有时你不得不面对它。

先来看一下system()函数的简单介绍:
#include <stdlib.h>
int system(const char *command);

system() executes a command specified in command by calling /bin/sh -c command, and returns after the command has been completed. During execution of the command, SIGCHLD will be blocked, and SIGINT and SIGQUIT will be ignored.

system()函数调用/bin/sh来执行参数指定的命令,/bin/sh 一般是一个软连接,指向某个具体的shell,比如bash,-c选项是告诉shell从字符串command中读取命令;
在该command执行期间,SIGCHLD是被阻塞的,好比在说:hi,内核,这会不要给我送SIGCHLD信号,等我忙完再说;
在该command执行期间,SIGINT和SIGQUIT是被忽略的,意思是进程收到这两个信号后没有任何动作。

再来看一下system()函数返回值:
The value returned is -1 on error (e.g. fork(2) failed), and the return status of the command otherwise. This latter return status is in the format specified in wait(2). Thus, the exit code of the command will be WEXITSTATUS(status). In case /bin/sh could not be executed, the exit status will be that of a command that does exit(127).
If the value of command is NULL, system() returns nonzero if the shell is available, and zero if not.
为了更好的理解system()函数返回值,需要了解其执行过程,实际上system()函数执行了三步操作:
1.fork一个子进程;
2.在子进程中调用exec函数去执行command;
3.在父进程中调用wait去等待子进程结束。
对于fork失败,system()函数返回-1。
如果exec执行成功,也即command顺利执行完毕,则返回command通过exit或return返回的值。
(注意,command顺利执行不代表执行成功,比如command:"rm debuglog.txt",不管文件存不存在该command都顺利执行了)
如果exec执行失败,也即command没有顺利执行,比如被信号中断,或者command命令根本不存在,system()函数返回127.
如果command为NULL,则system()函数返回非0值,一般为1.

看一下system()函数的源码
看完这些,我想肯定有人对system()函数返回值还是不清楚,看源码最清楚,下面给出一个system()函数的实现:
int system(const char * cmdstring)
{
    pid_t pid;
    int status;

if(cmdstring == NULL)
{
    return (1); //如果cmdstring为空,返回非零值,一般为1
}

if((pid = fork())<0)
{
    status = -1; //fork失败,返回-1
}
else if(pid == 0)
{
    execl("/bin/sh", "sh", "-c", cmdstring, (char *)0);
    _exit(127); // exec执行失败返回127,注意exec只在失败时才返回现在的进程,成功的话现在的进程就不存在啦~~
}
else //父进程
{
    while(waitpid(pid, &status, 0) < 0)
    {
        if(errno != EINTR)
        {
            status = -1; //如果waitpid被信号中断,则返回-1
            break;
        }
    }
}

    return status; //如果waitpid成功,则返回子进程的返回状态
}

仔细看完这个system()函数的简单实现,那么该函数的返回值就清晰了吧,那么什么时候system()函数返回0呢?只在command命令返回0时。

看一下该怎么监控system()函数执行状态
这里给我出的做法:
int status;
if(NULL == cmdstring) //如果cmdstring为空趁早闪退吧,尽管system()函数也能处理空指针
{
    return XXX;
}
status = system(cmdstring);
if(status < 0)
{
    printf("cmd: %s\t error: %s", cmdstring, strerror(errno)); // 这里务必要把errno信息输出或记入Log
    return XXX;
}

if(WIFEXITED(status))
{
    printf("normal termination, exit status = %d\n", WEXITSTATUS(status)); //取得cmdstring执行结果
}
else if(WIFSIGNALED(status))
{
    printf("abnormal termination,signal number =%d\n", WTERMSIG(status)); //如果cmdstring被信号中断,取得信号值
}
else if(WIFSTOPPED(status))
{
    printf("process stopped, signal number =%d\n", WSTOPSIG(status)); //如果cmdstring被信号暂停执行,取得信号值
}

到于取得子进程返回值的相关介绍可以参考另一篇文章:http://my.oschina.net/renhc/blog/35116

 

system()函数用起来很容易出错,返回值太多,而且返回值很容易跟command的返回值混淆。这里推荐使用popen()函数替代,关于popen()函数的简单使用也可以通过上面的链接查看。

popen()函数较于system()函数的优势在于使用简单,popen()函数只返回两个值:
成功返回子进程的status,使用WIFEXITED相关宏就可以取得command的返回结果;
失败返回-1,我们可以使用perro()函数或strerror()函数得到有用的错误信息。

这篇文章只涉及了system()函数的简单使用,还没有谈及SIGCHLD、SIGINT和SIGQUIT对system()函数的影响,事实上,之所以今天写这篇文章,是因为项目中因有人使用了system()函数而造成了很严重的事故。现像是system()函数执行时会产生一个错误:“No child processes”。

关于这个错误的分析,感兴趣的朋友可以看一下:http://my.oschina.net/renhc/blog/54582

 

2012-04-14 qdurenhongcai@163.com

转载请注明出处。

© 著作权归作者所有

恋恋美食

恋恋美食

粉丝 115
博文 170
码字总数 160207
作品 0
杭州
高级程序员
私信 提问
加载中

评论(10)

恋恋美食
恋恋美食 博主

引用来自“Jankinf”的评论

你好 为什么主进程要wait自己的pid呢
这就是system的实现机制呀。 system要等子进程结束后,才返回到外部调用。否则外边也不知道成功与否。
J
Jankinf
你好 为什么主进程要wait自己的pid呢
杨喜儿
杨喜儿
刚好用上,谢谢分析~
h
hk开源
分析的很透彻
恋恋美食
恋恋美食 博主

引用来自“ToakMa”的评论

博主你好,我现在遇到类似的问题。我的问如下:我通过sytem()来执行date -s "2015-03-01 12:00:00"也会导致我的程序奔溃;和你遇到问题不同的是,我在命令行输入date -s "2015-03-01 12:00:00"或者date -s 12:00:00时会奔溃,不知道你是否有相关经验?0

命令行崩溃?不明白
学跳舞的大象
学跳舞的大象
博主你好,我现在遇到类似的问题。我的问如下:我通过sytem()来执行date -s "2015-03-01 12:00:00"也会导致我的程序奔溃;和你遇到问题不同的是,我在命令行输入date -s "2015-03-01 12:00:00"或者date -s 12:00:00时会奔溃,不知道你是否有相关经验?0
perfectkuang
perfectkuang
谢谢咯!0
张瑞
呵呵,应该是我理解你的注释有误:)
恋恋美食
恋恋美食 博主

引用来自“张瑞”的评论

system()源码中26行的注释,应该是如果waitpid因为被SIGINT中断而返回-1,则需要重新waitpid,继续等待子进程; 如果waitpid是因为其他的原因而返回-1,则说明有系统失效,这时system()失败,返回-1。这样才符合POSIX.1规定的system在调用期间忽略SIGINT。

上面的写法是,如果waitpid()函数是被信号中断而返回负数的,则继续调用waitpid()函数。
这个包括SIGINT的啊,不违反POSIX.1定义啊
张瑞
system()源码中26行的注释,应该是如果waitpid因为被SIGINT中断而返回-1,则需要重新waitpid,继续等待子进程; 如果waitpid是因为其他的原因而返回-1,则说明有系统失效,这时system()失败,返回-1。这样才符合POSIX.1规定的system在调用期间忽略SIGINT。
【C/C++】Linux下system()函数引发的错误

今天,一个运行了近一年的程序突然挂掉了,问题定位到是system()函数出的问题,关于该函数的简单使用在我上篇文章做过介绍:http://my.oschina.net/renhc/blog/53580 先看一下问题 简单封装了...

恋恋美食
2012/04/21
20.4K
6
对于linux下system()函数的深度理解(整理)

对于linux下system()函数的深度理解(整理) (2013-02-07 08:58:54) 这几天调程序(嵌入式linux),发现程序有时就莫名其妙的死掉,每次都定位在程序中不同的system()函数,直接在shell下输入syste...

李东委
2014/11/30
402
0
用brk实现sbrk,关于brk的返回值

首先我们已经知道linux下,malloc最后调用的是sbrk函数,而sbrk是对brk的简单封装。 用sbrk模仿malloc很简单,sbrk(0)得到当前breakpoint,再调用sbrk(size)即可。(PS:breakpoint表示堆结束地...

不写可以么
2013/07/25
1K
0
Perl 程序在做多线程时 async 使用一定要谨慎

使用Perl开发Linux和Unix维护管理脚本是常用的手段,其中也会因为作业要求大量使用多线程技术。通常下我们使用 threads::create()创建一个新的线程,这时候往往我们会创建一个$thr来保存线程...

鉴客
2011/02/16
767
0
让你提高效率的 Linux 技巧

想要在 Linux 命令行工作中提高效率,你需要使用一些技巧。 巧妙的 Linux 命令行技巧能让你节省时间、避免出错,还能让你记住和复用各种复杂的命令,专注在需要做的事情本身,而不是你要怎么...

04%
2018/09/29
0
0

没有更多内容

加载失败,请刷新页面

加载更多

IT小白们进击前端工程师的学习路线:编辑器,基础进阶学习要点,框架

一、HTML、CSS基础、JavaScript语法基础。学完基础后,可以仿照电商网站(例如京东、小米)做首页的布局。 二、JavaScript语法进阶。包括:作用域和闭包、this和对象原型等。相信我,JS语法,...

梦想编程
8分钟前
19
0
ZhaoWei-2020-01-19

Dubbo Dubbo是一个分布式服务治理框架,提供高性能和透明化的RPC远程服务调用方案及 SOA架构治理方案。 远程通信 提供对多种基于长连接的NIO框架抽象封装,包括多种线程模型,序列化,以及 ...

SuSheePark
11分钟前
1
0
Python文件的常见标头格式是什么?

在有关Python编码准则的文档中,我遇到了以下Python源文件的头格式: #!/usr/bin/env python"""Foobar.py: Description of what foobar does."""__author__ = "Barack Obama"__cop......

javail
15分钟前
2
0
Linux 安装 jq

先下载jq安装包 https://stedolan.github.io/jq/download/将下载的安装包文件jq-linux64 拷贝到服务器下 wget -O jq https://github.com/stedolan/jq/releases/download/jq-1.6/jq-li......

乐易林谷
19分钟前
40
0
Elasticsearch深入:Refresh和Flush区别@

整体流程: 数据首先写入Buffer缓冲和Translog日志文件中。 当你写一条数据doc的时候,一方面写入到mem buffer缓冲中,一方面同时写入到translog日志文件中。 buffer满了或者每隔1秒(默认1秒...

HLee
23分钟前
4
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部