【问题标题】:How to run a program from terminal input(linux)?如何从终端输入(linux)运行程序?
【发布时间】:2016-04-08 14:05:10
【问题描述】:

虽然我认为我已经掌握了 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 程序做类似的事情吗?

【问题讨论】:

    标签: c linux


    【解决方案1】:

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

    实际原型是: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 个字符(减去终止的 null)不能拆分为生成超过 128 个单字符参数,每个参数与相邻的参数之间用一个单独的字符分隔空白字符。使用strtok() 是一种临时创可贴。一旦您需要处理真正的 shell 语法(管道、I/O 重定向、带空格的引号字符串等),strtok() 就是错误的工具。 (它——strtok()——也是在任何库函数中使用的错误函数。如果您必须在库代码中使用类似strtok() 的函数,请在 Unix 上使用 POSIX strtok_r() 或在 Windows 上使用 Microsoft 的 strtok_s()。)

    【讨论】:

    • 感谢您快速而彻底的回答。我无法完全理解的问题的唯一部分是假设我将 fileName 设置为一个值。你的意思是初始化?在那种情况下,我没有。我需要吗?在我看来,当程序运行时,它会提示用户,用户会输入类似“./helloWorld”或“/bin/./helloWorld”的内容。这将是 fileName 的值。
    • 输入名称就是我所说的“将fileName设置为一个值”。如所写,您使用未初始化的字符串作为命令,这不太可能正常工作。只要您将fileName 设置为某种稍微合适的东西,就应该没问题。 ./helloWorld/bin/./helloWorld 都可以工作,尽管 ./ 不是必需的(execv() 不受 $PATH 的影响,如果您仅指定 helloWorld,它将被视为与 ./helloWorld 相同) . /bin/./helloWorld 不确定; /./ 可能只是 /,您不应该将软件放入 /bin
    【解决方案2】:

    就目前而言,您需要修复代码中的编译错误:

    g++ -std=c++11 -g -Wall -Wextra -Wwrite-strings     36501711.cpp   -o 36501711
    36501711.cpp: In function ‘int main(int, char**)’:
    36501711.cpp:18:25: error: invalid conversion from ‘char’ to ‘const char*’ [-fpermissive]
             execv(fileName[0],fileName);
                             ^
    36501711.cpp:18:35: error: cannot convert ‘char*’ to ‘char* const*’ for argument ‘2’ to ‘int execv(const char*, char* const*)’
             execv(fileName[0],fileName);
                                       ^
    36501711.cpp: At global scope:
    36501711.cpp:7:5: warning: unused parameter ‘argc’ [-Wunused-parameter]
     int main(int argc, char* argv[])
         ^
    

    警告(关于未使用的argc)是无害的,但错误是真实的,需要修复。

    你需要在fork之前初始化fileName,否则孩子那里不会有任何有效数据:

    #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;
    
        //Infinite Loop
        while (1) {
            printf(" %s > ", argv[0]);
            scanf("%s", fileName);
            if (fileName[0] == '\0')
                break;
            printf("\n Entered file: %s\n", fileName);
    
            pid = fork();
            switch (pid) {
            case -1: //Did not fork properly
                perror("fork");
                break;
    
            case 0: //child
                execl(fileName, fileName, 0);
                perror("exec");
                exit(EXIT_FAILURE);
                break;
    
            default: //parent
                waitpid(pid,0,0); /* wait for child to exit() */
                break;
            }
        }
    
        return EXIT_SUCCESS;
    }
    

    【讨论】:

      最近更新 更多