写一个简单的 Linux Shell (C++)
这里可以找到代码
支持的特性
- 单条指令的执行
- 引号引起的参数(如
$ some_program "hello, world") - 重定向(>、< )
- 管道(|)
- 内建指令(如 cd、history、quit)
- 指令别名(如 ll → ls -l)
- 家目录(~)
运行截图

如何写一个简单的 Shell
这里简单介绍写 Shell 时比较关键的一些部分,具体请查看源代码。
展示提示符
见 show_command_prompt 函数。
command_prompt 是在每行最开始显示的一段与用户名、路径等相关的提示信息。ExpShell 显示的 prompt 形如 [root@localhost tmp]>。用 > 而非 #、$ 作为提示符,以区分原生 Shell。
获取用户名
passwd *pwd = getpwuid(getuid());
string username(pwd->pw_name);
获取当前目录
getcwd(char_buf, CHAR_BUF_SIZE);
string cwd(char_buf);
- prompt 中目录只显示最近一级,此处用
/来 split 后取最后一个即可 - 家目录需要折叠为
~,这里顺便把家目录地址存到全局变量home_dir,后续要用到
- prompt 中目录只显示最近一级,此处用
获取主机名
gethostname(char_buf, CHAR_BUF_SIZE);
string hostname(char_buf);
- 有时 hostname 会是形如
localhost.locald.xxx的形式,也 split 处理一下
- 有时 hostname 会是形如
输出之即可
cout << "[" << username << "@" << hostname << " " << cwd << "]> ";
解析命令
为存储解析结果,定义如下四个类:
- cmd:各种 cmd 的基类
- exec_cmd:形如
argv[0] argv[1] ...的普通命令 - pipe_cmd:管道命令,形如
left: cmd* | right: cmd* - redirect_cmd:重定向命令,形如
cmd_: cmd* > (or <) file
(最基础的)解析 exec_cmd
见
parse_exec_cmd函数。注意这里使用string_split_protect函数来 split 出 argv,这样可以保持被引号引起的带空格的 argument 不被拆分。解析一条命令
见
parse函数。采用分治法递归地解析命令。- 从左到右扫描字符串
- 如果是普通字符,则读入缓存
- 如果是重定向符号,将当前缓存解析为 exec_cmd,作为左手边 cmd;继续不断读入直到再次遇到符号或字符串结束,作为右手边 file,构建 redirect_cmd
- 如果是管道符号,递归地调用
parse解析右侧剩余,解析结果作为本层递归的右手边,构建 pipe_cmd
解析内建命令
主要支持 cd 、history 和 quit 命令。
- 调用
exit(0)即可实现 quit - history 命令根据记录打印即可
- 对于 cd,考虑如下情况
- 无参 cd 等价于
cd ~ - 对于形如
cd ~/some_path的命令,使用home_dir替换~ - 其他情况调用
chdir即可
- 无参 cd 等价于
- 调用
执行命令
主要见 run_cmd 函数。该函数接收一个 cmd*,递归地完成其链上所有 cmd 的执行。
对于 exec_cmd
- 检查别名,替换别名,例如 ll → ls -l
- 使用
execvp函数执行命令- 看 这篇博文 了解 exec 族函数,可见
execvp在当前场景最为合适 - 第二个参数是一个末元素为 NULL 的 char**(char*[]),内容为 argv
- 看 这篇博文 了解 exec 族函数,可见
对于 pipe_cmd
为 pipe_cmd 的 left 和 right 分别 fork 子进程执行,并使用管道让这两个兄弟进程通信
这张图很好地说明了父子进程使用管道通信的方法

依据上图,不难类比出兄弟进程进行 IPC 的方法如下
- 父进程 pipe
- 父进程 fork 两次
- child1 关读端,重定向写端,执行命令,关写端
- child2 关写端,重定向读端,执行命令,关读端
- 父进程关闭读、写端,并 wait
对于 redirect_cmd
- 打开 file,获得 fd
- 重定向 stdin 或 stdout 到 fd
- 执行命令
- 关闭 fd
主函数
在一个死循环中读入当前命令,如果不是 builtin_command,则 fork 子进程进行解析和执行(避免阻塞 ExpShell 自身),执行完成后子进程 exit。
其他细节
- pipe、open、dup2 等方法返回值小于 0 均表示出现错误,需要触发 panic
- 对于 wait 方法的状态字,当
WIFEXITED(status)为 0 时表示子进程异常退出,使用WEXITSTATUS(status)可以进一步获得子进程的 exit code
写一个简单的 Linux Shell (C++)的更多相关文章
- 写了一个简单的Linux Shell用来下载文件
#!/bin/sh ; i<; i=i+ )); do # 利用spider来探测请求的资源是否存在,并把请求的结果写入到一个文件 wget --spider --http-user=usern ...
- 实现一个简单的 Linux Shell(C++)
Implement a simple command interpreter in Linux. The interpreter should: support both internal and e ...
- 如何写一个简单的shell
如何写一个简单的shell 看完<UNIX环境高级编程>后我就一直想写一个简单的shell来作为练习,因为有事断断续续的写了好几个月,如今写了差不多来总结一下. 源代码放在了Github: ...
- 一个简单的linux下设置定时执行shell脚本的示例
很多时候我们有希望服务器定时去运行一个脚本来触发一个操作,比如说定时去备份服务器数据.数据库数据等 不适合人工经常做的一些操作这里简单说下 shell Shell俗称壳,类似于DOS下的command ...
- linux设备驱动第三篇:写一个简单的字符设备驱动
在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分 ...
- linux设备驱动第三篇:如何写一个简单的字符设备驱动?
在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存 ...
- 如何写一个简单的http服务器
最近几天用C++写了一个简单的HTTP服务器,作为学习网络编程和Linux环境编程的练手项目,这篇文章记录我在写一个HTTP服务器过程中遇到的问题和学习到的知识. 服务器的源代码放在Github. H ...
- 《Linux内核分析》第三周 构建一个简单的Linux系统MenuOS
[刘蔚然 原创作品转载请注明出处 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000] WEEK THREE ...
- 第三节 构造一个简单的Linux系统MenuOS——20135203齐岳
第三节 构造一个简单的Linux系统MenuOS By 20135203齐岳 Linux内核源代码 arch/ 支持不同cpu的源代码 Documentations/ 文档存储 init/ 内核启动相 ...
随机推荐
- C# 中生成随机数
生成1-100之间的随机数: new Random().Next(1, 100) 但是输出的结果是一样的,因为Random调用无参的构造函数其实调用的是 有参的构造函数,传递的 默认值: Enviro ...
- JavaScript学习系列博客_24_JavaScript 原型对象
原型(prototype) - 创建一个函数(所有函数)以后,解析器都会默认在函数中添加一个属性prototype prototype属性指向的是一个对象,这个对象我们称为原型对象. 创建一个函数My ...
- GRMS_README
基于Hadoop的商品推荐系统 基于特征:基于行为:具有了一定的历史特征. 基于用户: 基于商品: 推荐结果=用户的购买向量*物品的相似度矩阵 物品的相似度:物品的共现次数 1.项目名:GRMS2.添 ...
- win PHP7安装oracle扩展
环境介绍:win10.phpstudy php7.2版本 一.设置php.ini php.ini中开启相关扩展:extension=php_oci8_12c.dll.extension=php_pdo ...
- 2020.5.25 第五篇 Scrum冲刺博客
Team:银河超级无敌舰队 Project:招新通 项目冲刺集合贴:链接 目录 一.每日站立会议 1.1 会议照片 1.2 项目完成情况 二.项目燃尽图 三.签入记录 3.1 代码/文档签入记录 3. ...
- Federated Learning with Matched Averaging
挖个坑吧,督促自己仔细看一遍论文(ICLR 2020),看看自己什么时候也能中上那么一篇(流口水)~ 郑重声明:原文参见标题,如有侵权,请联系作者,将会撤销发布! Abstract 联邦学习允许边缘设 ...
- 简明python教程--读后感--推荐给python新手
原书名: A Byte of Python作者: Swaroop, C. H.译者: 沈洁元出版社: 未知 优点 1. 讲解很详细,很基础,适合入门,对编译器也做了简单的介绍 2. ...
- promise和async await的区别
在项目中第一次遇到async await的这种异步写法,来搞懂它 项目场景 :点击登录按钮后执行的事件,先进行表单校验 this.$refs.loginFormRef.validate(element ...
- Labview学习之路(七)for和while的理论要点
for循环 循环次数可以为0(N的接线端为) 终止条件:1. 完成N次循环. 2. 添加条件接线端,就像while循环的红点一样,(方法,右键点击边框,添加条件接线端) 数组通过自动索引接入 ...
- MySql密码的问题
由于长时间没使用过MySql了,也由于之前没有做笔记的习惯,晚上因为MySQL的密码问题导致数据库长时间没连上.纠结了这么久还是决定记录下来,毕竟安装的东西多了,这年头到处都是密码,加上时间一长,很容 ...