c - 用c编写shell时出现问题

标签 c linux shell fork

我正在尝试用c语言编写一个shell程序。 该程序需要具有通过 fork 函数创建的多个进程,并且能够在一个命令中打印多个输出行。 例如,像linux终端一样,如果输入是“ls ; ps ; pwd ;”,则输出应该是这样的。

$./shell
shell> ls ; ps ; pwd ;
(ls output)
(ps output)
(pwd output) 

它应该能够打开一个文件并显示命令列表和文件包含的输出。(我猜是批处理模式?)

假设这些命令列表位于批处理文件中。

batch
1 ls 
2 ps
3 ls ; pwd ; ps

输出是

$./shell batch
shell> ls
shell> (ls output)
shell> ps
shell> (ps output)
shell> ls ; pwd ; ps
shell> (ls output)
(ps output)
(pwd output)

这是我写的代码

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <string.h>
#include <fcntl.h>

void command(char* myargs[][10], char* buffer);
int tokenizing(char* myargs[][10], char* buffer);

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

    int fd;
    char buffer[200];
    char* myargs[10][10];
    char* token;

    if(argc >= 2) {
        if((fd=open(argv[1], O_RDONLY)) == -1)
            printf("cannot open file\n");
        else {
            read(fd, buffer, 200);
            printf("%s\n", buffer);
            token = strtok(buffer, "\n");
            while(token != NULL) {
                printf("%s\n", token);
                command(myargs, token);
                token = strtok(NULL, "\n");
            }
            return 0;
        }
    }



    while(1) {
        printf("prompt> ");
        if(fgets(buffer, 200, stdin) == NULL||
                strcmp(buffer, "quit\n") == 0)
            break;
        command(myargs, buffer);
    }

    return 0;
}

void command(char* myargs[][10], char* buffer) {

    int rc = fork();

    if(rc < 0) {
        fprintf(stderr, "fork failed\n");
    } else if(rc == 0) {

        int n = tokenizing(myargs, buffer);

        for(int i = 0 ; i < n; i++) {

            int rc2 = fork();
            if(rc2 < 0) {
                fprintf(stderr, "for failed\n");
            } else if(rc2 == 0) {
                execvp(myargs[i][0], myargs[i]);
                printf("%s: command not found\n", myargs[i][0]);
                exit(0);
            } else {
                wait(NULL);
            }
        }
        exit(0);
    }
    else {

        wait(NULL);


    }
}

int tokenizing(char* myargs[][10], char* buffer) {

    int i = 0;
    int j = 0;
    int k = 0;

    char* token;
    char* subCommand[10];

    token = strtok(buffer, ";\n");

    while(token != NULL) {
        subCommand[k] = token;
        k++;
        token = strtok(NULL, ";\n");
    }


    for(int i = 0; i < k; i++) {
        token = strtok(subCommand[i], " \n");
        while(token != NULL) {
            myargs[i][j] = token;
            j++;
            token = strtok(NULL, " \n");
        }
        myargs[i][j] = NULL;
        j=0;
    }
}

这段代码工作正常,但有一些问题。当此代码与批处理文件一起运行时,我遇到了一些错误。

enter image description here

当程序执行时,据我所知,我认为输出应该像上面的图像文件一样。

enter image description here

但程序经常会出现一些我什至没有输入的奇怪命令行。这些结果只是交替发生。

此外,如果您看到“ps”列表,则可以看到两个 shell 程序正在运行。

你们能帮我解决这些问题吗?

最佳答案

C 中的“字符串”需要 NUL 终止符。在未安全地以 NUL 终止的 char 数组上调用 printf("%s....") 和 strtok() 等 strng 函数会导致未定义的行为。

read() 返回一个值。您可以使用它将终止符加载到“缓冲区”中。为了 super 安全,您应该尝试只读取 [buffer size -1] 字符,以确保始终有足够的空间用于终止符,例如:

buffer[read(fd, buffer, 199)]='\0';

关于c - 用c编写shell时出现问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43015555/

相关文章:

计算完全替代的密码排列

c - 为什么这段代码读取第 4 个字符之后的所有内容都是 ' '?

python - 从 conda env 执行命令

c - 跟踪 malloc 分配了多少内存

c - 以所需的方式对数组进行排序

linux - ssh 上的 mercurial,远程 hg 没有合适的响应

c++ - Linux/CUPS 打印示例/教程?

c++ - gdb 调试器访问未找到的文件

shell - Scrapy Shell无法正常工作,无法使用命令

bash - 检查未传递的参数时出现意外标记 `fi' 附近的语法错误