linux 执行一个shell命令 并获取其返回值


#define RESULT_MAX_BUFF_SIZE  4096

int exec_cmd_and_get_result( const char *cmd_str, char *buffer ) 
{
    int cnt;
    FILE *pf;
    pf = popen(cmd_str, "r");
    cnt=fread(buffer, 1,RESULT_MAX_BUFF_SIZE, pf);
    buffer[cnt-1] = '\0';       

    
    printf( "fread cnt = %d\n", cnt );
    //printf( "strlen(buffer) = %d\n", strlen(buffer) );


    printf( "buffer = %s\n", buffer );
    
    pclose(pf);
    return 0;
}

void printdir(char *dir, int depth)
{
    DIR *dp;
    char buffer[RESULT_MAX_BUFF_SIZE];
    char cmd_str[RESULT_MAX_BUFF_SIZE];
    struct dirent *entry;
    struct stat statbuf;  
    if( (dp = opendir(dir)) == NULL ){
        fprintf(stderr,"cannot open directory: %s\n", dir);
        return;
    }
    //fprintf(stdout,"open directory: %s\n", dir);
    chdir(dir);
    while((entry = readdir(dp)) != NULL) {
        lstat(entry->d_name,&statbuf);
        if( S_ISDIR(statbuf.st_mode) ){
            if( strcmp(".",entry->d_name) == 0 || strcmp("..",entry->d_name) == 0 ){
                continue;
            }
            //printf("%*s%s/\n",depth,"",entry->d_name);
            if( strcmp("sys",entry->d_name) == 0 || strcmp("proc",entry->d_name) == 0 ){
                continue;
            }
            printdir(entry->d_name,depth+4);
        }else{
            //printf("%*s%s\n",depth,"",entry->d_name);
            
            exec_cmd_and_get_result("pwd", buffer);
             //printf("result = %s\n", buffer);

             strcat(buffer, "/");
             strcat(buffer, entry->d_name);

             printf("path = %s\n", buffer);


             strcpy( cmd_str, "cat ");
             strcat(cmd_str, buffer);

             exec_cmd_and_get_result(cmd_str, buffer);
             //printf("result = %s\n", buffer);
               
             //printf("result1 = %s\n", buffer);

             if(strlen(buffer)==83){
                 printf("result1 = %s\n", buffer);


             }

             printf("result1 len = %d\n", strlen(buffer) );
             
             
        }
    }
    chdir("..");
    closedir(dp);
}

<think>好的,我现在需要解决用户的问题:如何在使用system()函数执行命令获取命令返回值。用户之前的问题是关于grep和排除PID返回值的处理,现在转向了C语言中的system()函数,这可能意味着他们在编写脚本或程序时需要处理子进程的退出状态。 首先,我得回忆一下system()函数的基本用法。system()是C标准库中的一个函数,用于执行一个shell命令。它的声明在stdlib.h头文件中,函数原型是int system(const char *command)。这个函数会启动一个子进程来运行指定的命令,然后返回命令执行结果状态。 接下来,用户想知道如何获取命令返回值。这里需要注意的是,system()返回的不是被调用命令的直接返回值,而是一个状态码,这个状态码需要经过宏处理才能得到实际的退出状态。比如,使用WEXITSTATUS宏来提取子进程的退出码。 可能的误区是,用户可能认为system()的返回值就是命令的退出码,但实际上返回值包含更多信息,比如信号中断等。因此,正确处理返回值是关键。例如,如果子进程正常退出,可以用WEXITSTATUS来获取退出码;如果被信号终止,则需要用WTERMSIG来获取信号编号。 用户可能在使用system()时遇到了问题,比如无法正确判断命令是否成功执行。例如,当执行一个返回非零命令时,他们可能没有正确解析system()的返回值,导致逻辑错误。这时候需要解释如何正确使用宏来分解返回值。 此外,用户可能有跨平台的需求,因为不同系统对system()的实现可能略有不同,但WEXITSTATUS等宏是POSIX标准的,因此在大多数Unix-like系统上可用。如果在Windows环境下,可能需要不同的方法,但用户的问题中没有提到,暂时假设是Unix环境。 可能的示例代码会有帮助。例如,编写一个简单的C程序,使用system()运行ls命令,然后使用WIFEXITED判断是否正常退出,再用WEXITSTATUS获取退出码。如果返回0,表示成功,非零则表示错误。 还要注意错误处理的情况,比如system()本身可能失败,例如无法启动shell,此时返回值可能是-1。这时候需要检查errno来确定具体错误原因。 总结用户的真实需求:他们可能正在编写一个需要执行外部命令检查其执行结果的程序,需要了解如何正确地从system()的返回值中提取命令的实际退出状态,以便进行后续的错误处理或逻辑判断。需要详细解释步骤,提供示例代码,确保他们能够正确应用。</think>--- ### **问题定位** 使用 C/C++ 中的 `system()` 函数执行命令时,如何正确获取命令返回值?直接通过 `system()` 的返回值无法直接对应命令的退出码,需通过 **宏定义** 和 **状态码分解** 处理。 --- ### **核心原理** #### **1. `system()` 函数的行为** - **功能**:调用 `/bin/sh -c "command"` 执行命令。 - **返回值类型**:`int`,表示命令的 **综合状态**(非直接退出码)。 - **返回值规则**: - 如果 `system()` 启动 shell 失败(如内存不足),返回 `-1`。 - 如果命令成功执行返回值为 **命令的退出状态编码**(需解析)。 #### **2. 状态码的组成** 返回值一个 16 位整数,包含两部分: - **高 8 位**:实际命令的退出码(`0~255`)。 - **低 8 位**:信号编号(若进程被信号终止)。 --- ### **正确获取退出码的步骤** #### **步骤 1:判断 `system()` 是否成功** ```c int status = system("ls /tmp"); if (status == -1) { // system() 调用失败(如无法启动 shell) perror("system() error"); exit(1); } ``` #### **步骤 2:解析命令的退出状态** 使用宏定义分解状态码: - `WIFEXITED(status)`:检查进程是否正常退出(非信号终止)。 - `WEXITSTATUS(status)`:提取实际退出码(仅当 `WIFEXITED` 为真时有效)。 ```c if (WIFEXITED(status)) { int exit_code = WEXITSTATUS(status); printf("命令退出码:%d\n", exit_code); } else { printf("进程被信号终止,信号编号:%d\n", WTERMSIG(status)); } ``` --- ### **完整示例代码** ```c #include <stdio.h> #include <stdlib.h> #include <sys/wait.h> int main() { const char *command = "ls /nonexistent"; // 故意执行一个会失败的命令 int status = system(command); if (status == -1) { perror("system() 执行失败"); return 1; } if (WIFEXITED(status)) { int exit_code = WEXITSTATUS(status); if (exit_code == 0) { printf("命令执行成功\n"); } else { printf("命令执行失败,退出码:%d\n", exit_code); } } else { printf("进程被信号终止,信号编号:%d\n", WTERMSIG(status)); } return 0; } ``` #### **输出结果** ``` ls: 无法访问 '/nonexistent': 没有那个文件或目录 命令执行失败,退出码:2 ``` --- ### **常见错误场景与修复** | 现象 | 原因 | 修复方法 | |-----------------------|-------------------------------|-----------------------------------| | 直接使用 `system()` 返回值作为退出码 | 未解析高 8 位状态码 | 使用 `WEXITSTATUS()` 宏提取退出码 | | 未处理信号终止的情况 | 未检查 `WIFEXITED()` 的返回值 | 先判断进程是否正常退出 | | 忽略 `system()` 自身的失败 | 未检查 `status == -1` | 添加错误处理逻辑 | --- ### **跨平台注意事项** - **POSIX 标准**:`WEXITSTATUS`、`WIFEXITED` 等宏定义在 `<sys/wait.h>` 中,适用于 Linux/macOS。 - **Windows 差异**:Windows 的 `system()` 返回值直接为命令的退出码,无需解析。但需注意: ```c int status = system("dir"); printf("退出码:%d\n", status); // Windows 下直接输出退出码 ``` --- ### **总结** - **核心步骤**: 1. 检查 `system()` 是否成功调用。 2. 使用 `WIFEXITED()` 判断进程是否正常退出。 3. 通过 `WEXITSTATUS()` 提取实际退出码。 - **关键点**:理解状态码的二进制结构,避免直接使用 `system()` 的原始返回值
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值