1

我正在尝试将字符串拆分为标记以创建参数参数数组。我当前的实现如下(path是正在读取可选参数的用户可执行文件的路径):

// ARG_MAX as defined in limits.h
int execute(char *exe) {
    printf("args to %s: ", exe);

    char *args = malloc(ARG_MAX);
    scanf("%s", args);

    char *argv[ARG_MAX];

    int i = 0;
    argv[i++] = exe;

    while ((argv[i] = strsep(&args, " \t")) != NULL) {
        i++;
    }

    free(args);
    execv(exe, argv);
    return 0;
}

令我困惑的是,根据我的理解,strsep这应该按预期工作,并且在一定程度上,在测试时它准确地分配为tokens[0]be path,并且直到第一个空白字符为止。tokens[1]tokens_s

但是,当您在空格后输入另一个参数时,这不会分配到tokens[2],以此类推用于后续参数。

我似乎无法发现在使用它时我做错了什么strsep并没有导致所需的功能?

输入: exe = "/usr/bin/ps" args = "-e -l"

输出: exec ps -e

4

1 回答 1

4

多个错误:

  • 您必须阅读参数fgets()以阅读多个单词。

  • 您必须使用临时变量,strsep()以便可以将原始指针从malloc()back 传递到free(),或者简单地使用本地数组。

这是一个更正的版本:

#include <errno.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <limits.h>

// ARG_MAX as defined in limits.h
int execute(char *exe) {
    char args[ARG_MAX + 1];

    printf("args to %s: ", exe);
    fflush(stdout);
    if (fgets(args, sizeof args, stdin)) {
        char *argv[ARG_MAX / 2];
        char *p;

        int i = 0;
        argv[i++] = exe;

        p = args;
        args[strcspn(args, "\n")] = '\0';  // strip the newline if present
        while ((argv[i] = strsep(&p, " \t")) != NULL) {
            i++;
        }

        printf("argv: ");
        for (i = 0; argv[i]; i++)
            printf(" '%s'", argv[i]);
        printf("\n");

        execv(exe, argv);
        printf("exec failed: %s\n", strerror(errno));
    } else {
        printf("cannot read input\n");
    }
    return 0;
}

int main(int argc, char *argv[]) {
    char *exe = "printf";
    if (argc > 1)
        exe = argv[1];
    return execute(exe);
}

笔记:

  • execv如果成功,将不会返回您的程序。

  • strsep不会折叠分隔符序列,如果您有额外的空格,您的方法将创建额外的参数。

编辑:stdin如果在运行之前读取输入execute,并且如果通过调用执行此类输入,则缓冲区scanf()中可能有一个挂起的换行符,并将其读取为空行。如果是这种情况,请在调用之前先刷新待处理的输入:stdinfgets()printf()

int c;
while ((c = getchar()) != EOF && c != '\n') {
    continue;
}
于 2016-11-15T23:30:45.143 回答