如何在 Linux 中使用参数执行 C 代码中的外部程序?

2024-10-11 08:36:00
admin
原创
265
摘要:问题描述:我想在 C 代码中执行另一个程序。例如,我想执行一个命令./foo 1 2 3 foo是存在于同一文件夹中的程序,并且1 2 3是参数。foo程序创建一个将在我的代码中使用的文件。我该如何做?解决方案 1:简单的方法是使用system():#include <stdlib.h> ...

问题描述:

我想在 C 代码中执行另一个程序。例如,我想执行一个命令

./foo 1 2 3

foo是存在于同一文件夹中的程序,并且1 2 3是参数。
foo程序创建一个将在我的代码中使用的文件。

我该如何做?


解决方案 1:

简单的方法是使用system()

#include <stdlib.h>
...
int status = system("./foo 1 2 3");

system()将等待 foo 完成执行,然后返回一个状态变量,您可以使用它来检查例如 exitcode(该命令的 exitcode 乘以 256,因此将 system() 的返回值除以该值即可获得实际的 exitcode:)int exitcode = status / 256

的手册页wait()(在第 2 部分,man 2 wait在您的 Linux 系统上)列出了您可以用来检查状态的各种宏,其中最有趣的是WIFEXITEDWEXITSTATUS

或者,如果您需要读取 foo 的标准输出,请使用popen(3),它返回一个文件指针(FILE *);与命令的标准输入/输出交互与读取或写入文件相同。

解决方案 2:

system函数调用 shell 来运行命令。虽然这很方便,但众所周知它存在安全隐患。如果您可以完全指定要执行的程序或脚本的路径,并且可以承受失去所system提供的平台独立性,那么您可以使用下面函数execve中所示的包装器exec_prog来更安全地执行程序。

以下是在调用者中指定参数的方法:

const char    *my_argv[64] = {"/foo/bar/baz" , "-foo" , "-bar" , NULL};

然后exec_prog像这样调用该函数:

int rc = exec_prog(my_argv);

该函数如下exec_prog

static int exec_prog(const char **argv)
{
    pid_t   my_pid;
    int     status, timeout /* unused ifdef WAIT_FOR_COMPLETION */;

    if (0 == (my_pid = fork())) {
            if (-1 == execve(argv[0], (char **)argv , NULL)) {
                    perror("child process execve failed [%m]");
                    return -1;
            }
    }

#ifdef WAIT_FOR_COMPLETION
    timeout = 1000;

    while (0 == waitpid(my_pid , &status , WNOHANG)) {
            if ( --timeout < 0 ) {
                    perror("timeout");
                    return -1;
            }
            sleep(1);
    }

    printf("%s WEXITSTATUS %d WIFEXITED %d [status %d]
",
            argv[0], WEXITSTATUS(status), WIFEXITED(status), status);

    if (1 != WIFEXITED(status) || 0 != WEXITSTATUS(status)) {
            perror("%s failed, halt system");
            return -1;
    }

#endif
    return 0;
}

记住包括:

#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <stdio.h>

有关需要通过文件描述符(如和)与执行程序进行通信的情况,请参阅相关的 SE 帖子。stdin`stdout`

解决方案 3:

您可以使用fork()并且system()这样您的程序就不必等到system()返回。

#include <stdio.h>
#include <stdlib.h>

int main(int argc,char* argv[]){

    int status;

    // By calling fork(), a child process will be created as a exact duplicate of the calling process.
    // Search for fork() (maybe "man fork" on Linux) for more information.
    if(fork() == 0){ 
        // Child process will return 0 from fork()
        printf("I'm the child process.
");
        status = system("my_app");
        exit(0);
    }else{
        // Parent process will return a non-zero value from fork()
        printf("I'm the parent.
");
    }

    printf("This is my main program and it will continue running and doing anything i want to...
");

    return 0;
}

解决方案 4:

system()执行 shell,然后 shell 负责解析参数并执行所需程序。要直接执行程序,请使用 fork() 和 exec()(system() 使用这些函数执行 shell,shell 本身也使用这些函数执行命令)。

#include <unistd.h>

int main() {
     if (fork() == 0) {
          /*
           * fork() returns 0 to the child process
           * and the child's PID to the parent.
           */
          execl("/path/to/foo", "foo", "arg1", "arg2", "arg3", 0);
          /*
           * We wouldn't still be here if execl() was successful,
           * so a non-zero exit value is appropriate.
           */
          return 1;
     }

     return 0;
}

解决方案 5:

在 C 中

#include <stdlib.h>

system("./foo 1 2 3");

在 C++ 中

#include <cstdlib>

std::system("./foo 1 2 3");

然后像平常一样打开并读取文件。

解决方案 6:

这样怎么样:

char* cmd = "./foo 1 2 3";
system(cmd);

解决方案 7:

当您没有对参数进行硬编码时,这里有扩展到变量参数的方法(尽管在这个例子中它们在技术上仍然是硬编码的,但应该很容易弄清楚如何扩展......):

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int argcount = 3;
const char* args[] = {"1", "2", "3"};
const char* binary_name = "mybinaryname";
char myoutput_array[5000];

sprintf(myoutput_array, "%s", binary_name);
for(int i = 0; i < argcount; ++i)
{
    strcat(myoutput_array, " ");
    strcat(myoutput_array, args[i]);
}
system(myoutput_array);
相关推荐
  政府信创国产化的10大政策解读一、信创国产化的背景与意义信创国产化,即信息技术应用创新国产化,是当前中国信息技术领域的一个重要发展方向。其核心在于通过自主研发和创新,实现信息技术应用的自主可控,减少对外部技术的依赖,并规避潜在的技术制裁和风险。随着全球信息技术竞争的加剧,以及某些国家对中国在科技领域的打压,信创国产化显...
工程项目管理   2079  
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   1459  
  建筑行业正处于数字化转型的关键时期,建筑产品生命周期管理(PLM)系统的实施对于提升项目效率、质量和协同性至关重要。特别是在 2025 年,基于建筑信息模型(BIM)的项目进度优化工具成为众多建筑企业关注的焦点。这些工具不仅能够整合项目全生命周期的数据,还能通过精准的分析和模拟,为项目进度管理提供强大支持。BIM 与建...
plm是什么软件   0  
  PLM系统开发的重要性与现状PLM(产品生命周期管理)系统在现代企业的产品研发、生产与管理过程中扮演着至关重要的角色。它贯穿产品从概念设计到退役的整个生命周期,整合了产品数据、流程以及人员等多方面的资源,极大地提高了企业的协同效率和创新能力。通过PLM系统,企业能够实现产品信息的集中管理与共享,不同部门之间可以实时获取...
国产plm软件   0  
  PLM(产品生命周期管理)系统在企业产品研发与管理过程中扮演着至关重要的角色。随着市场竞争的加剧和技术的飞速发展,企业对PLM系统的迭代周期优化需求日益迫切。2025年敏捷认证对项目管理提出了新的要求,其中燃尽图作为一种强大的可视化工具,在PLM系统迭代周期优化中有着广泛且重要的应用。深入探讨这些应用,对于提升企业的项...
plm系统主要干什么的   0  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用