Linux之模拟shell命令行解释器

简介: Linux之模拟shell命令行解释器

前言

本文是基于前面介绍过的关于进程创建、进程终止、进程等待、进程替换等知识,尝试做的一个简单的shell命令解释器。


一、输出提示符

1.实际

2.模拟

printf("用户名@主机名 当前路径#");
fflush(std);

此处没有“\n”,会有缓冲区的问题,因此要用fflush(std);来刷新缓冲区

二、输入指令、获取指令

1.实际

2.模拟

  1. 输入
char lineCommand[NUM];
char* s = fgets(lineCommand,sizeof(lineCommand) - 1,stdin);
assert(s != NULL);
lineCommand[strlen(linCommand) - 1] = 0;

fgets函数获取一行的内容,将内容存在字符数组lineCommand中。

因为fgets获取也会将enter作为'\n’获取,为了避免多打印一行,我们要将最后一个元素重置为'\0'

  1. 获取
strtok(lineCommand," ");

用strtok函数将输入的字符串切割成若干个子串;

strtok函数的参数:第一次传字符串的首地址,之后传NULL即可(会从上次切割的位置继续切割)。

三、fork创建子进程

利用fork创建子进程;

替换函数需要带上v和p,即execvp函数。

为啥要用子进程去执行命令?

答:如果不创建子进程,而是让bash直接去执行进程,会将我们的bash直接替换为其他程序,shell就不能继续正常执行其他指令了(就回不到输入界面了)。

四、内建命令

我们在运行自己写的shell,输入cd ../cd path等命令时,发现路径并没有发生修改,这是为啥?

答:因为我们自己写的shell,执行很多命令都要用fork创建子进程,让子进程去执行。当子进程执行cd命令时,更改的时子进程的工作目录而与父进程无关,因此父进程的路径并不会发生修改。

因此,对于cd命令我们应该用内建命令:该命令不需要子进程执行,而是让bash自己执行。要修改程序的工作目录需要用chdir系统调用。

什么是当前路径?

当前路径就是cwd。

cwd -> 当前进程所在的工作目录(默认是文件存储在磁盘的路径);

exe -> 当前运行的是磁盘路径下的哪个进程。

更改当前进程的工作目录:chdir。(谁调用我,我就更改谁的工作目录)

五、代码实现

1 #include<stdio.h>
  2 #include<assert.h>
  3 #include<string.h>
  4 #include<unistd.h>
  5 #include<sys/types.h>
  6 #include<sys/wait.h>
  7 #include<stdlib.h>
  8 #define NUM 1024
  9 #define OPT_NUM 64
 10 char lineCommand[NUM];
 11 char* myargv[OPT_NUM];
 12 int lastCode = 0;
 13 int lastSig = 0;
 14 int main()
 15 {
 16         while(1)
 17         {
 18         printf("用户名@主机名 路径#");
 19         fflush(stdout);
 20         char* s = fgets(lineCommand,sizeof(lineCommand) - 1, stdin);
 21         assert(s != NULL);
 22         (void)s; 
 23         lineCommand[strlen(lineCommand) - 1] = 0;
 24         myargv[0] = strtok(lineCommand," "); 
 25         int i = 1;
 26         if(myargv[0] != NULL && strcmp(myargv[0], "ls") == 0)
 27         {
 28                 myargv[i++] = (char*)"--color=auto";
 29         }       
 30         while(myargv[i++] = strtok(NULL," "));
 31         if(myargv[0] != NULL && strcmp(myargv[0], "cd") == 0)
 32         {
 33                 if(myargv[1] != NULL) chdir(myargv[1]);
 34                 continue;
 35         }       
 36         if(myargv[0] != NULL && strcmp(myargv[0], "echo") == 0)
 37         {
 38                 if(strcmp(myargv[1], "$?") == 0)
 39                 {
 40                         printf("%d %d\n", lastCode, lastSig);
 41                 }       
 42                 else
 43                 {
 44                         printf("%s\n",myargv[1]);
 45                 }
 46                 continue;
 47         }
 48         #ifdef DEBUG
 49         for(int i = 0; myargv[i]; ++i)
 50         {
 51                 printf("myargv[%d]=%s\n", i, myargv[i]);
 52         }
 53         #endif
 54         pid_t id = fork();
 55         if(id < 0)
 56         {
 57                 perror("fork");
 58                 exit(1);
 59         }
 60         else if(id == 0)
 61         {
 62                 //
 63                 execvp(myargv[0], myargv);
 64                 exit(1);
 65         }
 66         int status = 0;
 67         pid_t ret = waitpid(id, &status, 0);
 68         assert(ret > 0);
 69         (void)ret;
 70         lastSig = WIFEXITED(status);
 71         lastCode = WEXITSTATUS(status);
 72         }
 73         return 0;
 74 }

运行:

文件tt.c


总结

以上就是今天要讲的内容,本文介绍了如何实现一个简单的shell解释器。本文作者目前也是正在学习Linux相关的知识,如果文章中的内容有错误或者不严谨的部分,欢迎大家在评论区指出,也欢迎大家在评论区提问、交流。

最后,如果本篇文章对你有所启发的话,希望可以多多支持作者,谢谢大家!

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