Как выполнить внешнюю программу в C-коде в Linux с помощью аргументов?
Я хочу выполнить другую программу в коде C.
Например, я хочу выполнить команду
./foo 1 2 3
foo
- это программа, которая существует в той же папке, и 1 2 3
являются аргументами.
foo
программа создает файл, который будет использоваться в моем коде.
Как это сделать?
Ответы
Ответ 1
Используйте system()
:
int status = system("./foo 1 2 3");
system()
будет ожидать завершения выполнения foo, а затем вернуть переменную состояния, которую вы можете использовать для проверки, например. ExitCode. man 2 wait
в вашей системе linux будут перечислены различные макросы, которые вы можете использовать для проверки состояния, наиболее интересными будут WIFEXITED
и WEXITSTATUS
В качестве альтернативы, если вам нужно прочитать вывод foo в stdout, используйте popen()
.
Ответ 2
Функция system
вызывает оболочку для запуска команды. Хотя это удобно, оно хорошо известно последствия для безопасности. Если вы можете полностью указать путь к программе или script, который вы хотите выполнить, и вы можете позволить себе потерять независимость платформы, которую предоставляет 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]\n",
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.\n");
status = system("my_app");
exit(0);
}else{
// Parent process will return a non-zero value from fork()
printf("I'm the parent.\n");
}
printf("This is my main program and it will continue running and doing anything i want to...\n");
return 0;
}
Ответ 4
В C
#include <stdlib.h>
system("./foo 1 2 3");
В С++
#include <cstdlib>
std::system("./foo 1 2 3");
Затем откройте и прочитайте файл, как обычно.
Ответ 5
#include <unistd.h>
int main() {
if (fork() == 0) {
execl("foo", "foo", "1", "2", "3", 0);
# We woundn't still be here if execl() was successful, so we exit with a non-zero value.
return -1;
}
return 0;
}
Ответ 6
Как об этом:
char* cmd = "./foo 1 2 3";
system(cmd);
Ответ 7
Здесь вы можете перейти к переменным args, когда у вас нет жестких кодировок args (хотя в этом примере они по-прежнему технически жестко закодированы, но должно быть легко выяснить, как расширить...):
#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);