Shell 的简单实现

环境

选项 参数
系统 Windows 10 下 VMware Workstation 虚拟机 ubuntu 14.04.6 desktop i386
GCC gcc version 4.8.4 (Ubuntu 4.8.4-2ubuntu1~14.04.4)

功能

  1. 解析并执行用户提交的命令行
  2. 提供 lsmkdirrmdirpwdps 等内部命令(多进程并使用 execvp 函数调用)
  3. 最后一个参数为 & 时,命令后台执行,同时父进程应当等待子进程
  4. 提供历史查询功能:当用户按下 Ctrl+C 时输出最近的 10 个命令行
  5. 执行历史命令行:当用户输入 r x 时,执行首字母为 x 的最近的命令行,如果没有参数则执行最近的命令
  6. 用户输入 Ctrl+D 时退出程序

步骤

Ctrl+C

实验要求在用户按下 Ctrl+C 时,不退出程序,而改为输出最近输入的 10 条命令,首先需要捕获,然后重新定义它的功能。

void my_handle(int sig);

int main()
{
    signal(SIGINT, my_handle);
    /* ... */
}

void my_handle(int sig)
{
    int i, j;
    printf("\nCaught Control C\n");
    if (idx == -1)
        printf("Never entered a command.\n");
    else
        for (i = 0; i < MAX_HISTORY; ++i)
            if (!history[j = (idx - i) < 0 ? idx - i + MAX_HISTORY : idx - i][0])
                break;
            else
                printf("%2d: %s\n", i + 1, history[j]);
}

发现 Ctrl+C 的确变成了输出最近输入的 10 条命令,但是在输出完之后依旧会退出,如下图

最后发现 Ctrl+C 每次发出 SIGINT 信号,但是在 SIGINT 信号处理完之后,其处理函数会重置,不再是 my_handle,所以在每次调用 my_handle 时都需要重新设置,作如下更改:

void my_handle(int sig)
{
    signal(SIGINT, my_handle);
    /* ... */
}

更改完之后如下

发现缺少 COMMAND-> 提示,重新把它加上。

Ctrl+D

实验要求在用户按下 Ctrl+D 时退出程序,由于 Ctrl+D 并非如 Ctrl+C 一般发送一个信号,而是作为一个特殊的字符。在键盘缓冲区内有数据时, Ctrl+D 会刷新缓冲区,使程序能读取到这些字符;在键盘缓冲区内没有字符时,Ctrl+D 会向程序发送 EOF,使用 getc 等函数读取时会返回 -1,所以需要自行实现 readline 类的输入函数并判断是否键入 Ctrl+D

输入并解析

由于存储的历史记录为完整的命令,除了在输入之后需要解析之外,在使用 r x 命令时也需要再次解析,所以选择将解析的功能剥离出 setup

首先使用 getc 函数逐个读取字符,直接使用 scanf 等函数不易判断 EOF

/**
 * 读取命令并调用 parse_command 解析命令
 *
 * @param buffer 输入的命令
 * @param args 存放解析完的参数
 * @param background 为 1 时表示后台执行
 **/
void setup(char buffer[], char* args[], int* background)
{
    /**
     * i 表示命令中当前字符的下标
     * j 表示参数个数
     * k 用于查找历史记录
     **/
    int i, j, k;
    // 读取命令
    i = 0;
    do {
        // 读取一字符
        buffer[i] = getc(stdin);
        // 捕获 Ctrl+D
        if (buffer[i] == EOF) {
            printf("\nCaught Control D\n");
            exit(0);
        }
    } while (buffer[i++] != '\n');
    // 将 '\n' 结尾的字符串替换为 '\0' 结尾的标准字符串
    buffer[--i] = '\0';
    // 解析命令
    j = parse_command(buffer, args);
    /* 后续处理 */
}

对命令行进行解析,以空白符为分隔:

/**
 * 解析命令
 *
 * @param buffer 输入的命令
 * @param args 存放解析完的参数
 *
 * @return 参数个数
 **/
int parse_command(char buffer[], char* args[])
{
    /**
     * i 表示命令中当前字符的下标
     * j 表示命令中当前参数的下标
     * k 表示命令中当前参数的长度
     * c 用于读取的字符并处理
     **/
    int i, j, k;
    char c;
    // 指针置 NULL 但不释放
    i = 0;
    while (args[i])
        args[i++] = NULL;
    // 分割参数
    j = k = 0;
    for (i = 0; i < MAX_SIZE; ++i)
        if (isspace(buffer[i]) || !buffer[i]) {
            if (i && !isspace(buffer[i - 1])) {
                /**
                 * 分配空间并拷贝字符串
                 * k + 1 保留 '\0' 的空间
                 **/
                args[j] = malloc(sizeof(char) * (k + 1));
                // 标识字符串末尾
                c = buffer[i];
                buffer[i] = '\0';
                // 拷贝参数
                strcpy(args[j++], buffer + i - k);
                buffer[i] = c;
                k = 0;
            }
            if (!buffer[i]) break;
        } else ++k;
    // 返回参数个数
    return j;
}

由于 r x 命令出现在历史记录中可能导致产生命令死循环,所以禁止其加入历史记录:

void setup(char buffer[], char* args[], int* background)
{
    /* 输入命令 */
    // 解析命令
    j = parse_command(buffer, args);
    // 如果输入了一条非空命令
    if (j) {
        if (strcmp("r", args[0]))
            // 保存命令,FIFO
            strcpy(history[idx = (idx + 1) % MAX_HISTORY], buffer);
        else if (idx == -1)
            // 没有历史记录
            printf("Never entered a command.\n");
        else {
            k = idx;
            // 执行历史中记录的命令
            if (args[1] && args[1][0])
                // 查找命令
                for (i = 0; i < MAX_HISTORY; ++i)
                    if (args[1][0] == history[k = (idx - i) < 0 ? idx - i + MAX_HISTORY : idx - i][0])
                        break;
            // 释放空间
            i = 0;
            while (args[i])
                free(args[i++]);
            //解析命令
            j = 0;
            if (k == MAX_HISTORY)
                printf("Command not found.\n");
            else
                j = parse_command(history[k], args);
        }
        // 后台执行
        if (j && !strcmp("&", args[j - 1])) {
            // 设置 background 值
            *background = 1;
            --j;
        }
    }
    // 以 NULL 标识参数的结束
    args[j] = NULL;
}

main 函数,实现整体功能:

int main()
{
    char buffer[MAX_SIZE];
    /**
     * 存储参数
     * 
     * (MAX_SIZE + 1) / 2 保证能存储所有参数
     * 1 保证有多余的空位存储 NULL 以标识参数的结尾
     **/
    char* args[(MAX_SIZE + 1) / 2 + 1];
    int background, i;
    pid_t pid;
    // 捕获 Ctrl+C
    signal(SIGINT, my_handle);
    memset(history, 0, sizeof(history));
    while (1) {
        background = 0;
        printf("COMMAND-> ");
        // 读取并解析命令
        setup(buffer, args, &background);
        pid = fork();
        if (pid == -1) {
            fprintf(stderr, "fork error\n");
            // 释放空间
            i = 0;
            while (args[i])
                free(args[i++]);
        } else if (pid == 0) {
            execvp(args[0], args);
        } else if (background) {
            pid = wait(NULL);
            printf("[parent] child process %d done\n", pid);
        }
    }
}

父进程等待

测试中发现功能基本已经完善,但是在输入后台命令时,只有执行第一条后台命令时父进程会等待子进程,之后的命令父进程都不会等待,如下图

分析 wait 函数发现只要有一个子进程完成,wait 函数就会返回,所以父进程不会等待之后的子进程。

wait 函数更改为 waitpid,等待指定的子进程:

pid = waitpid(pid, NULL, 0);

最终结果如下

至此完成所有功能。

实验总结

本次实验主要学习了如何使用多进程来完成工作,因为 execvp 会替换当前进程,所以如果不使用多进程将会导致每次执行只能解析运行一条指令。同时了解了如何设置专门的信号处理函数,信号在编程中如何使用,也了解到了 Ctrl+CCtrl+D 的区别。

个人博客:https://wilfredshen.cn/

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351