当前位置: 首页 > backend >正文

自主 Shell 命令行解释器

我们制作的目标是要求自制的命令行解释器能够处理普通命令,能够处理内建命令,理解本地变量,环境变量。

我们的 shell 命令是由 bash 创建子进程来执行的,这样可以更好的保护操作系统。同理,我们在写命令解释器时,用 fork 子进程的方式来完成指令,父进程走主线,是一个比较合理的方式。

如图这就是我们要实现的一个大体思路。 

那么在 shell 命令行中应当要包括哪些内容呢?

[root@localhost epoll]# ls
client.cpp readme.md server.cpp utility.h
[root@localhost epoll]# ps
PID TTY TIME CMD
3451 pts/0 00:00:00 bash
3514 pts/0 00:00:00 ps

 我们需要下面这个循环过程,获取命令行,解析命令行,建立一个子进程,用execvp替换子进程,父进程等待子进程退出。

下面我们就来实现一个shell了!

首先我们来捋清一下思路

首先我们通过数组来存储指令信息,当每次处理完一次指令之后,需要对数组进行初始化;初始化完数组后我们要获取当前的用户信息并将其打印下来,” 用户名@当前路径:家目录# ” 的方式打印出来;接下来我们需要获取用户输入的指令并且解析,以空格为分隔符将指令存入数组当中;最后是对指令进行分支处理,若指令为内键命令则执行内键分支,若不为内键命令就创建子进程执行命令。最后在循环返回。

 一. 初始化数据

首先我们在 shell.cc 中创建全局变量 ,数组char *gargv,和数组计数器gargc,数组pwd记录输入的指令,最后是退出码lastcode。

我们从系统环境变量中获取USERNAME ,HOSTNAME ,PWD,HOMEPATH。

初始化时将存储指令的数组 gargv进行清0.

#include "shell.h"
#include <iostream>
#include <cstdio>
#include <cstring>
#include <string.h>
#include <ctype.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <string>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
char *gargv[ARGS] = {NULL};
int gargc = 0;
char pwd[1024];
int lastcode = 0;static std::string GetUserName(){std::string username = getenv("USER");return username.empty() ? "None" : username;}static std::string GetHostName(){std::string hostname = getenv("HOSTNAME");return hostname.empty() ? "None" : hostname;}static std::string GetPwd(){char temp[1024];getcwd(temp,sizeof(temp));//将temp写入pwd,并且更新内容snprintf(pwd,sizeof(pwd),"PWD=%s",temp);putenv(pwd);std::string pwd_lable = temp;const std::string pathsep = "/";auto pos = pwd_lable.rfind(pathsep);if(pos == std::string::npos){                                                                                                                   return "None";}pwd_lable = pwd_lable.substr(pos+pathsep.size());return pwd_lable.empty() ? "/" : pwd_lable;}static std::string GetHomePath(){std::string home = getenv("HOME");return home.empty() ? "/" : home;}void InitGlobal(){gargc = 0;memset(gargv,0,sizeof(gargv));}

我们通过getcwd的方式从环境变量中获得相应的值。在GetPwd处,我们将temp写入pwd中,重新更新环境变量的值。从temp的尾部找到第一个“/”处返回“/”后的所有字符。

二. 获取用户信息并打印

仿照我们Linux的输出方式,” 用户名@当前路径:家目录# ”输出即可。

void PrintCommandPrompt(){std::string user = GetUserName();std::string hostname = GetHostName();std::string pwd = GetPwd();printf("[%s@%s %s]# ",user.c_str(),hostname.c_str(),pwd.c_str());}

三. 获取用户指令

将用户输入的指令传入数组中,并把最后的回车符号去除。

bool GetCommandString(char cmd_str_buff[],int len){if(cmd_str_buff == NULL || len <= 0)return false;char *res = fgets(cmd_str_buff,len,stdin);if(res == NULL)return false;cmd_str_buff[strlen(cmd_str_buff)-1]=0;return strlen(cmd_str_buff)==0 ? false : true;}

四. 解析用户指令

以空格为分隔符,用strtok将指令一一分离。

bool ParseCommandString(char cmd[]){if(cmd == NULL)return false;#define SEP " "gargv[gargc++] = strtok(cmd,SEP);while((bool)(gargv[gargc++]=strtok(NULL,SEP)));gargc--;return true;}

五. 子进程执行命令or执行内键命令

fork创建子进程,用execvp执行指令。内键命令分为两个大分支,cd和echo,在cd下若有两个字符,当为“~”时,跳转到家目录,其他则更改目录到 gargv【1】处。若只有一个字符,跳转到家目录。若为echo时,有两个字符,第一个字符为 “$”,第二个字符为“?”时,打印最近一次退出码;若第二个字符不是“?”则打印环境变量。若第一个字符不为“$”,打印值即可。

 bool BuiltInCommandExec()
{std::string cmd = gargv[0];bool ret = false;if(cmd == "cd"){if(gargc == 2){std::string target = gargv[1];if(target == "~"){ret = true;chdir(GetHomePath().c_str());}else{ret = true;chdir(gargv[1]);}}                                                                                          else if(gargc == 1){ret = true;chdir(GetHomePath().c_str());}else{// BUG()}}else if(cmd == "echo"){if(gargc == 2){std::string args = gargv[1];if(args[0] == '$'){if(args[1] == '?'){printf("lastcode:%d\n",lastcode);lastcode = 0;ret = true;}else{const char *name = &args[1];printf("%s\n",getenv(name));lastcode = 0;ret = true;}}else{printf("%s\n",args.c_str());ret = true;}}}return ret;}

希望各位大佬多多支持!!!

http://www.xdnf.cn/news/14025.html

相关文章:

  • Dify创建 echarts图表 (二)dify+python后端flask实现
  • [MSPM0开发]之七 MSPM0G3507 UART串口收发、printf重定向,解析自定义协议等
  • 如何解决答题小程序大小超过2M的问题
  • C#使用ExcelDataReader高效读取excel文件写入数据库
  • 华为云Flexus+DeepSeek征文|基于华为云一键部署 Dify 应用的性能测试实践:构建聊天应用并使用 JMeter做压力测试
  • HarmonyOS5 运动健康app(一):健康饮食(附代码)
  • 苹果获智能钱包专利,Find My生态版图或再扩张:钱包会“说话”还能防丢
  • 【论文阅读笔记】ICLR 2025 | 解析Ref-Gaussian如何实现高质量可交互反射渲染
  • pom文件引用外部jar依赖
  • Web开发实战:Gin + GORM 构建企业级 API 项目
  • 使用 C/C++ 和 OpenCV 判断是否抬头
  • Spring 事务传播行为详解
  • 自己的服务器被 DDOS跟CC攻击了怎么处理,如何抵御攻击?
  • 公司内网远程访问配置教程:本地服务器(和指定端口应用)实现外网连接使用
  • 29-Oracle 23ai Flashback Log Placement(闪回日志灵活配置)
  • 进程控制
  • Trae Builder 模式:从需求到全栈项目的端到端实践
  • 书写时垂直笔画比水平笔画表现更好的心理机制分析
  • Android binder内核漏洞研究(一)——环境搭建
  • 【MySQL基础】表的约束的类型与使用指南
  • 从Apache OFBiz 17.12.01的反序列化漏洞到Docker逃逸的渗透之红队思路
  • GaussDB 分布式数据库调优(架构到全链路优化)
  • C#实战:解决NPOI读取Excel单元格数值0.00001显示为1E-05的问题
  • [特殊字符] Harmony OS Next里的Web组件:网页加载的全流程掌控手册
  • macOS 查看当前命令行的ruby的安装目录
  • 大语言模型的分类与top3
  • Spark 之 Subquery
  • matlab实现非线性Granger因果检验
  • 深度学习:PyTorch张量基本运算、形状改变、索引操作、升维降维、维度转置、张量拼接
  • 将后端数据转换为docx文件