gpt4 book ai didi

c - 如何从终端输入(linux)运行程序?

转载 作者:IT王子 更新时间:2023-10-29 00:37:39 25 4
gpt4 key购买 nike

虽然我认为我掌握了 fork()exec()wait()pid 在 C 中工作,我还没有找到如何从程序中运行个人程序的方法。

这是我的代码:

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h> /* for fork() */
#include<sys/types.h> /* for pid_t */
#include<sys/wait.h> /* fpr wait() */

int main(int argc, char* argv[])
{
char fileName[255];
pid_t pid;

switch (pid = fork()) {
case -1: //Did not fork properly
perror("fork");
break;

case 0: //child
execv(fileName[0],fileName);

puts("Oh my. If this prints, execv() must have failed");
exit(EXIT_FAILURE);
break;
default: //parent
//Infinite Loop
while (1) {
printf(" %s > ", argv[0]);
scanf("%s", fileName); // gets filename
if (fileName[0] == '\0') continue;
printf("\n Entered file: %s",fileName); // prints the fileName
waitpid(pid,0,0); /* wait for child to exit() */
break;
}
}

return 0;
}

我的问题如下:

  1. 我想将一个字符串作为输入,并将其范围限制为 255 个字符。 char fileName[255] 然后 scanf("%s", fileName); 是要走的路吗?我应该改用 getLine() 还是其他函数?

  2. 假设输入正确。我如何执行现有的 hello world 程序。输入会存储在 *argv[] 中吗?我发现在不同的程序中我可以使用

    static char *argv[] = { "echo", "Foo is my name." , NULL };
    execv("/bin/echo", argv);

    为了回应“Foo is my name.”。我可以用 helloWorld 程序做类似的事情吗?

最佳答案

您传递的是单个字符作为命令名称,名称字符串作为参数列表的开头——就好像 execv() 的原型(prototype)是 int execv(char cmd, char *args).

实际原型(prototype)为:int execv(char *cmd, char **args),所以需要:

char *args[2] = { fileName, 0 };

execv(args[0], args);

我假设您在某处将 fileName 设置为一个有意义的值——未显示。例如,它可能是 "./local_program"。它将被视为可执行文件的路径名。

如果你想读取名称,那么你可以使用fgets()getline(),但你需要删除换行符:

if (fgets(fileName, sizeof(fileName), stdin) != 0)
{
fileName[strcspn(fileName, "\n")] = '\0';
…as before…
}

或:

char *fileName = 0;
size_t length = 0;
if (getline(&fileName, &length, stdin) != -1) /* Not EOF! */
{
fileName[strcspn(fileName, "\n")] = '\0';
…as before…
}
free(fileName);

strcspn() 的使用避免了必须对过长的命令行进行特殊处理,以至于 fileName 中没有换行符。请注意,这不会尝试将输入行拆分为命令名称和空格处的参数或类似的东西。这是下一级的 shell 实现:

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

int main(void)
{
char fileName[256];
if (fgets(fileName, sizeof(fileName), stdin) != 0)
{
fileName[strcspn(fileName, "\n")] = '\0';
char *args[129];
char **argv = args;
char *cmd = fileName;
const char *whisp = " \t\f\r\b\n";
/* Ugh — strtok()? OK while only handling white space separators */
char *token;
while ((token = strtok(cmd, whisp)) != 0)
{
*argv++ = token;
cmd = 0;
}
*argv = 0;

execv(args[0], args);
fprintf(stderr, "Oops!\n");
}
return 1;
}

我不需要检查 args 数组是否溢出,因为 256 个输入字符(减去终止空值)无法拆分以产生超过 128 个单字符参数,每个参数都与相邻的一个空白字符。使用 strtok() 是一个临时的创可贴。只要您需要处理真正的 shell 语法(管道、I/O 重定向、带空格的引号字符串等),strtok() 就很不幸地是错误的工具。 (它 — strtok() — 也是任何库函数中使用的错误函数。如果必须使用 strtok(),请在 Unix 上使用 POSIX strtok_r() 或在 Windows 上使用 Microsoft 的 strtok_s()类似于库代码中的函数。)

关于c - 如何从终端输入(linux)运行程序?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36501711/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com