Linux之简单的Shell命令行解释器

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: Linux之简单的Shell命令行解释器



我们前面学习了进程创建,进程终止,进程等待,进程替换,通过这些内容我们可以来进行实现简单的shell命令行解释器。下面我们直接来看一看如何去实现shell命令行解释器。

一、基本原理

1、我们知道,在打开shell解释器后,除非我们自己关闭它,那么它将一直不退出,所以命令行解释器一定是一个死循环。

2、输入:我们需要获取命令行上一行的内容,利用fgets函数获取,同时,可以定义一个cmd_line[NUM]数组来保存用户输入的内容。

3、解析:输入之后,我们自然需要去进行字符串的解析,我们需要分割字符串,将其分成命令+选项等内容,这个地方用strtok函数,把字符串切割成若干个子串,并将分开的各个子串保存至char* g_val[SIZE]数组中。strtok:第一次直接传递参数,第二次则必须传NULL。且在最终strtok会返回NULL。

4、我们在输入命令后,发现界面还是维持的原样,这是因为父进程会创建子进程去帮助其实现命令,同时父进程需要等待子进程退出返回结果。

5、子进程在执行命令的时候,我们需要使用进程替换的知识,将代码换成命令执行的代码。

6、有些命令,如cd命令,如果我们是交给子进程去完成的话,只会改变子进程的路径,所以cd命令必须由父进程亲自来完成。

不需要创建子进程执行,让shell自己执行命令,称为内建命令。本质就是执行系统接口,我们可以调用一个系统接口chdir,可以让父进程自己改变路径。

7、我们在输入完命令后,通常会按回车建使其执行,而fgets则会将它当成输入的一个字符,在给用户呈现结果时会多换一行,而不会紧挨着上一行去显示结果。所以我们要执行下面的操作:

cmd_line[strlen(line_cmd)-1] = '\0';

8、再比如,像ll这种命令,其等效于ls -l,因此我们需要将其在数组char* g_val[SIZE]中换成 g_val[0] = "ls",g_val[1] = "-l"即可。

二、代码实现

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<sys/wait.h>
#include<sys/types.h>
#include<unistd.h>
#define NUM 32
#define SIZE 1024
#define SEP " "
char cmd_line[NUM];//存放输入的字符串
char* g_val[SIZE];//存放解析出来的字符串
int main()
{
    //命令行解释器是一个不退出的进程
    while(1)
    {
       printf("[zdl@hcss-ecs-d14f ~]# ");
       fflush(stdout);
       memset(cmd_line, '\0', sizeof(cmd_line));
       if(fgets(cmd_line,sizeof(cmd_line),stdin) == NULL)
       {
           continue;
       }
       cmd_line[strlen(cmd_line)-1] = '\0';
       //命令行字符解析
       g_val[0] = strtok(cmd_line, SEP);
       int index = 1;
       if(strcmp(g_val[0],"ll")==0)
       {
           g_val[0]="ls";
           g_val[index++]="-l";
       }
       while(g_val[index++] = strtok(NULL, SEP));//如果还解析原字符串,就传入NULL
       //让父进程执行的命令叫做内置命令
       if(strcmp(g_val[0], "cd")==0)
       {
           if(g_val[1] != NULL)
               chdir(g_val[1]);
           continue;
       }
       //创建子进程去执行
       pid_t id = fork();
       if(id==0)
       {
           execvp(g_val[0], g_val);
           exit(1);
       }
       //父进程
       int status = 0;
       pid_t ret = waitpid(id, &status,0);
       if(ret > 0)
       {
           printf("exit code: %d\n",WEXITSTATUS(status));
       }
    }
    return 0;
}

代码运行结果如下:

目录
相关文章
|
28天前
|
域名解析 网络协议 安全
在Linux中,想在命令行下访问某个网站,并且该网站域名还没有解析,如何做?
在Linux中,想在命令行下访问某个网站,并且该网站域名还没有解析,如何做?
|
15天前
|
Shell Linux 开发工具
linux shell 脚本调试技巧
【9月更文挑战第3天】在Linux中调试shell脚本可采用多种技巧:使用`-x`选项显示每行命令及变量扩展情况;通过`read`或`trap`设置断点;利用`echo`检查变量值,`set`显示所有变量;检查退出状态码 `$?` 进行错误处理;使用`bashdb`等调试工具实现更复杂调试功能。
|
20天前
|
安全 Linux 开发工具
探索Linux操作系统:从命令行到脚本编程
【8月更文挑战第31天】在这篇文章中,我们将一起潜入Linux操作系统的海洋,从最基础的命令行操作开始,逐步深入到编写实用的脚本。无论你是初学者还是有一定经验的开发者,这篇文章都将为你提供新的视角和实用技能。我们将通过实际代码示例,展示如何在日常工作中利用Linux的强大功能来简化任务和提高效率。准备好了吗?让我们一起开启这段旅程,探索Linux的奥秘吧!
|
20天前
|
Linux
探索Linux操作系统:命令行与脚本编程基础
【8月更文挑战第31天】在这篇文章中,我们将一起踏上一段旅程,深入探索Linux操作系统的奥秘。通过学习命令行的使用和编写简单的脚本,你将能够更高效地与你的计算机进行交流。无论你是新手还是有经验的用户,本文都将为你打开一扇通往Linux世界的大门。准备好了吗?让我们开始吧!
|
21天前
|
运维 监控 Linux
深入理解Linux系统运维:命令行与脚本的奥秘
【8月更文挑战第30天】在Linux的世界里,命令行是运维人员的灵魂之窗。掌握命令行,就像握住了一把钥匙,能开启系统管理的宝藏箱。本文将带你走进Linux的命令行世界,通过实际代码示例,解锁那些高效管理和维护系统的秘籍。你将学到不仅仅是命令本身,更是如何将这些命令编织成强大的脚本,让日常的运维工作变得游刃有余。准备好跟随我的步伐,一起深入探索Linux命令行与脚本的奥秘吧!
|
22天前
|
Linux
Linux命令行文档查看cat、less、more、head、tail和图片查看
Linux命令行文档查看cat、less、more、head、tail和图片查看
41 0
|
28天前
|
监控 Shell Linux
在Linux中,如何使用shell脚本检测磁盘使用率?
在Linux中,如何使用shell脚本检测磁盘使用率?
|
28天前
|
Shell Linux 开发工具
在Linux中,如何编写shell脚本将当前目录下大于10K的文件转移到/tmp目录下?
在Linux中,如何编写shell脚本将当前目录下大于10K的文件转移到/tmp目录下?
|
28天前
|
缓存 Shell Linux
在Linux中,bash shell 中的 hash 命令有什么作用?
在Linux中,bash shell 中的 hash 命令有什么作用?
|
29天前
|
Unix Shell Linux
在Linux中,什么是Shell脚本,并且如何编写它。
在Linux中,什么是Shell脚本,并且如何编写它。