基于c++实现的简易shell

代码逻辑

核心思想

  • 解析命令行,拆解命令及其选项
  • 创建子进程,在子进程中执行命令
  • 如果是前台执行命令,则父进程就阻塞等待子进程中命令执行结束后回收子进程的资源
  • 如果是后台执行命令,则父进程不进行阻塞等待,可继续向下运行,但为了防止僵尸进程的出现,需要设置一个信号函数,当后台子进程的命令执行结束后,会给父进程发送一个SIGCHLD信号,父进程的信号函数收到该信号后知道子进程执行结束了,于是执行waitpid系统函数回收子进程

小知识

僵尸进程

  • 每个进程都有一个父进程,当该进程执行结束后,其资源需要被父进程回收,否则会占用系统资源。
  • 父进程回收子进程的办法就是通过调用waitpid函数阻塞等待子进程的执行结束。
  • 僵尸进程就是指当一个进程结束之后,父进程由于一些原因迟迟不回收其子进程的资源,从而导致该进程一直占用着系统资源,这时该进程就成为僵尸进程

比如,当一个父进程由于业务原因需要一直运行,那么由他所创建的子进程结束之后,如果不进行回收就会成为一个僵尸进程,而该父进程由于业务原因也不可能阻塞等待子进程执行结束之后回收子进程,此时就需要设置信号函数来回收进程

孤儿进程

与僵尸进程对应的还有一个概念叫孤儿进程

前面我们说,每个进程都一个父进程,负责将其所创建的子进程执行结束之后回收子进程的资源,但前提是,父进程需要活到子进程执行结束的时候,因此如果该子进程执行结束之前,父进程先挂掉了,这种情况下的子进程就叫孤儿进程

但孤儿进程的危害其实并不大,尽管父进程挂掉了,但是还有祖宗收场,因此,系统会将孤儿进程挂到进程号为0,也就是init进程下,由他负责回收孤儿进程的资源

信号

  • 信号是进程间通信的一种机制,通常用来通知进程发生了某种事件,如中断、错误或特定条件的满足。
  • 信号可以被发送到进程以提醒其执行特定操作,例如终止、暂停或处理错误。
  • 常见的信号包括 SIGINT(中断信号)、SIGTERM(终止信号)等。通过信号,程序可以对系统事件进行响应,提高灵活性和控制能力。

本文中的shell代码中涉及到了SIGCHLD信号的使用,该信号用于子进程结束或停止时,通知父进程其子进程的状态改变。父进程可以通过捕获此信号来调用 wait()waitpid() 函数,以获取子进程的退出状态并回收资源。这有助于防止僵尸进程的产生,确保资源得到有效管理。

源码

main.cc

#include <iostream>
#include <string>
constexpr int CMDMAXSIZE = 1024;
void eval(const char *cmdline);
void shell() {char cmdline[CMDMAXSIZE] = {0};while (1) {std::cout << ">";// 读取stdin输入的命令行if (!fgets(cmdline, sizeof(cmdline), stdin)) {// 判断是否输入到文件尾if (feof(stdin))exit(0);}// 解析命令行并执行命令eval(cmdline);}
}int main() {shell();return 0;
}

eval.cc

该文件用于解析命令后执行命令

#include <iostream>
#include <signal.h>
#include <string>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
constexpr int ARGSIZE = 1024;//解析命令并将其存储到argv,并返回该命令是一个后台命令
int parseline(const char *command, char **argv);
// 判断是否是一个shell内置命令,如果是就直接执行,否则返回false
int inlinecommand(char **argv);
// 后台回收进程
void backprogrom(int sig);void eval(const char *cmdline) {if (!cmdline)return;char *argv[ARGSIZE] = {0};// 解析命令并判断是否是后台命令int bg = parseline(cmdline, argv);if (bg == -1)return;// 判断是否是shell内置命令,如果是就直接执行,并返回标志int flag = inlinecommand(argv);if (flag == -1)return;pid_t pid;// 判断是否是shell内置命令,如果是就处理内置命令,否则说明是一个可执行文件if (!flag) {// 无论是不是后台命令,都执行可执行文件if ((pid = fork()) == 0) {if (execve(argv[0], argv, environ) < 0) {std::cerr << "exec failed!" << std::endl;_exit(1);}} else if (pid > 0) {if (!bg) {// 如果不是后台命令,父进程需要阻塞等待子进程(也就是命令行)的执行wait(nullptr);} else if (bg > 0) { //如果是后台命令,父进行不需要阻塞等待子进程执行结束std::cout << "pid [" << pid << "] " << argv[0] << " running..."<< std::endl;signal(SIGCHLD, backprogrom);}}}
}

parseline.cc

该文件负责解析命令行及其选项,并返回判断是否为后台命令

#include <sstream>
#include <string>
#include <vector>
int parseline(const char *command, char **argv) {std::stringstream cmdline(command);std::vector<std::string> argv_vec;std::string cmd;while (cmdline >> cmd) {argv_vec.emplace_back(cmd);}if (argv_vec.empty())return -1;int argc = 0;for (auto &str : argv_vec) {argv[argc++] = const_cast<char *>(str.c_str());}if (argv_vec[argv_vec.size() - 1] == "&") {argv[argc - 1] = nullptr;return 1;}argv[argc] = nullptr;return 0;
}

inlinecommand.cc

该文件用于判断输入的命令是否是shell的内置命令,如果是就直接执行之

#include <iostream>
#include <stdlib.h>
#include <string>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <unordered_set>
constexpr int CMDMAXSIZE = 1024;static std::unordered_set<std::string> inlinecmd = {"ls", "pwd", "clear"};int inlinecommand(char **argv) {std::string cmd(argv[0]);if (cmd == "exit") {exit(1);}auto it = inlinecmd.find(cmd);if (it != inlinecmd.end()) {pid_t pid = fork();if (pid == 0) {if (execvp(cmd.c_str(), argv) < 0) {std::cout << argv[0] << " exec error!" << std::endl;}}return 1;}return 0;
}

backwait.cc

该文件用于回收后台命令所在的子进程的资源

#include <iostream>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
/*
1. exit
库:exit 定义在 <stdlib.h> 中。
功能:exit 在程序终止时会执行以下操作:调用所有已注册的 atexit 函数。刷新标准 I/O 缓冲区,确保所有数据都被写入。返回控制给操作系统,返回值为 exit 的参数。
适用场景:通常用于正常退出程序。
2. _exit
库:_exit 定义在 <unistd.h> 中。
功能:_exit 直接终止进程,不执行任何清理操作:不调用 atexit 函数。不刷新 I/O 缓冲区。直接将控制权返回给操作系统,返回值为 _exit 的参数。
适用场景:通常在子进程中使用,如在fork后立即退出,以避免在父进程和子进程之间共享未刷新缓冲的数据。
*/
ssize_t sio_puts(const char s[]) { return write(STDOUT_FILENO, s, strlen(s)); }
void sio_error(const char s[]) {sio_puts(s);return _exit(1);
}void backprogrom(int sig) {int olderrno = errno;while (waitpid(-1, nullptr, WNOHANG) > 0) {sio_puts("reaped child\n");}if (errno != ECHILD)sio_error("wait_pid error!");errno = olderrno;
}

测试效果

说明,本代码仅用于帮助理解shell执行命令的过程,但是对后台命令的处理稍显不足,并且对前台命令和后台命令的回收处理也仍旧需要改进,烦请大佬帮忙指正!

参考:《深入理解计算机系统》

【Linux】fork函数详解|多进程_多进程fork-CSDN博客

孤儿进程与僵尸进程[总结] - Rabbit_Dale - 博客园 (cnblogs.com)

 ​​​​​​孤儿进程与僵尸进程产生及其处理_孤儿进程的意义-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1542978.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

【机器学习】---神经架构搜索(NAS)

这里写目录标题 引言1. 什么是神经架构搜索&#xff08;NAS&#xff09;1.1 为什么需要NAS&#xff1f; 2. NAS的三大组件2.1 搜索空间搜索空间设计的考虑因素&#xff1a; 2.2 搜索策略2.3 性能估计 3. NAS的主要方法3.1 基于强化学习的NAS3.2 基于进化算法的NAS3.3 基于梯度的…

【数据结构】图的遍历

快乐的流畅&#xff1a;个人主页 个人专栏&#xff1a;《C游记》《进击的C》《Linux迷航》 远方有一堆篝火&#xff0c;在为久候之人燃烧&#xff01; 文章目录 引言一、深度优先遍历1.1 定义1.2 实现 二、广度优先遍历2.1 定义2.2 实现 三、DFS与BFS的对比 引言 前置知识&…

linux用户管理运行级别找回root密码

目录 1.用户的添加 1.1用户添加的基本指令 1.2不指定家目录的名称 1.3指定家目录的名称 2.密码的修改 3.删除目录 3.1删除的两个情况 3.2删除的流程 4.查询用户的信息 5.用户的切换 6.用户组 6.1用户组的概念 6.2创建用户到指定的组 6.3修改用户到其他的组 6.4用…

SpringCloud Alibaba之Sentinel实现熔断与限流

&#xff08;学习笔记&#xff09; QPS&#xff08;Query Per Second&#xff09;&#xff1a;即每秒查询率&#xff0c;是对⼀个特定的查询服务器在规定时间内所处理流量多少的衡量标准。QPS req/sec 请求数/秒&#xff0c;即每秒的响应请求数&#xff0c;也即是最⼤吞吐能⼒…

ATTCK实战系列-Vulnstack三层网络域渗透靶场(一)

ATT&CK实战系列-Vulnstack三层网络域渗透靶场&#xff08;一&#xff09; 一、环境搭建1.1 靶场拓扑图1.2 靶场下载链接1.3 虚拟机配置1.3.1 Windows 7 (web服务器)1.3.2 Windows 2008 (域控)1.3.3 Win2k3 (域内主机) 二、外网打点突破2.1 信息搜集2.2 phpmyadmin 后台 Get…

肾癌的多模态预测模型-临床-组织学-基因组

目录 摘要 技术路线 ① lncRNA的预测模型 ②病理 WSI 的分类器 ③临床病理分类器 模型结果 与别的模型比较 同行评审学习 1&#xff09;使用lncRNA的原因 2&#xff09;模型临床使用意义 3&#xff09;关于截止值的使用 摘要 A multi-classifier system integrated…

.NET常见的5种项目架构模式

前言 项目架构模式在软件开发中扮演着至关重要的角色&#xff0c;它们为开发者提供了一套组织和管理代码的指导原则&#xff0c;以提高软件的可维护性、可扩展性、可重用性和可测试性。 假如你有其他的项目架构模式推荐&#xff0c;欢迎在文末留言&#x1f91e;&#xff01;&a…

Java_Day04学习

类继承实例 package com.dx.test03; public class extendsTest {public static void main(String args[]) {// 实例化一个Cat对象&#xff0c;设置属性name和age&#xff0c;调用voice()和eat()方法&#xff0c;再打印出名字和年龄信息/********* begin *********/Cat cat ne…

实战OpenCV之直方图

基础入门 直方图是对数据分布情况的图形表示&#xff0c;特别适用于图像处理领域。在图像处理中&#xff0c;直方图通常用于表示图像中像素值的分布情况。直方图由一系列矩形条&#xff08;也被称为bin&#xff09;组成&#xff0c;每个矩形条的高度表示某个像素值&#xff08;…

鸿蒙设置,修改APP图标和名称

1、先看默认的图标和名称 2、打开项目开始设置自己需要的图标和名称 2.1找到 路径src\main\module.json5&#xff0c; 找到 abilities&#xff0c;下的&#xff0c;图标icon、名称label&#xff0c;label可以按住ctrl鼠标左键点击跳转 2.2先修改APP名称 1、ctrl鼠标左键点击…

华为OD机试 - 选修课(Python/JS/C/C++ 2024 E卷 100分)

华为OD机试 2024E卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试真题&#xff08;Python/JS/C/C&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;私信哪吒&#xff0c;备注华为OD&#xff0c;加入华为OD刷题交流群&#xff0c;…

【C语言零基础入门篇 - 15】:单链表

文章目录 单链表链表的基本概念单链表功能的实现单链表的初始化单链表新结点的创建单链表头插法单链表的输出单链表的查找单链表修改单链表的删除单链表所有数据结点释放源代码 单链表 链表的基本概念 一、什么是链表&#xff1f; 链表是数据结构中线性表的一种&#xff0c;其…

华为OD机试 - 需要打开多少监控器(Java 2024 E卷 100分)

华为OD机试 2024E卷题库疯狂收录中&#xff0c;刷题点这里 专栏导读 本专栏收录于《华为OD机试&#xff08;JAVA&#xff09;真题&#xff08;E卷D卷A卷B卷C卷&#xff09;》。 刷的越多&#xff0c;抽中的概率越大&#xff0c;私信哪吒&#xff0c;备注华为OD&#xff0c;加…

软考高级:数据库保持函数依赖和有损无损分解 AI 解读

讲解 生活化例子 想象你经营着一家快餐店&#xff0c;店里有各种商品&#xff0c;你也记录了每天的销量。你有一个表格&#xff0c;记录了「商品名称」、「价格」、「库存数量」、「供应商信息」等数据。最开始&#xff0c;你可能把所有数据都写在一张表上&#xff0c;但时间…

2024年9月22日---关于MyBatis框架(1)

一 Mybatis概述 1.1 简介 MyBatis&#xff08;官网&#xff1a;mybatis – MyBatis 3 | 简介 &#xff09;是一款优秀的开源的 持久层 框架&#xff0c;用于简化JDBC的开发。是 Apache的一个开源项目iBatis&#xff0c;2010年这个项目由apache迁移到了google code&#xff0c…

PCL 随机下采样

目录 一、概述 1.1原理 1.2实现步骤 1.3应用场景 二、代码实现 2.1关键函数 2.2完整代码 三、实现效果 PCL点云算法汇总及实战案例汇总的目录地址链接&#xff1a; PCL点云算法与项目实战案例汇总&#xff08;长期更新&#xff09; 一、概述 随机下采样 是一种常用的点…

类和对象(2)(重点)

个人主页&#xff1a;Jason_from_China-CSDN博客 所属栏目&#xff1a;C系统性学习_Jason_from_China的博客-CSDN博客 所属栏目&#xff1a;C知识点的补充_Jason_from_China的博客-CSDN博客 类的默认成员函数 概念概述 默认成员函数就是用户没有显式实现&#xff0c;编译器会自…

项目扩展一:信道池的实现

项目扩展一&#xff1a;信道池的实现 一、为何要设计信道池1.引入信道的好处2.为何要设计信道池 二、信道池的设计1.服务器需要设计信道池吗&#xff1f;2.设计&#xff1a;动态变化的信道池1.为什么&#xff1f;2.怎么办&#xff1f;1.动态扩容和缩容2.LRU风格的信道置换3.小总…

0基础学习HTML(十三)布局

HTML 布局 网页布局对改善网站的外观非常重要。 请慎重设计您的网页布局。 如何使用 <table> 元素添加布局。 网站布局 大多数网站会把内容安排到多个列中&#xff08;就像杂志或报纸那样&#xff09;。 大多数网站可以使用 <div> 或者 <table> 元素来创建…

软件测试分类篇(下)

目录 一、按照测试阶段分类 1. 单元测试 2. 集成测试 3. 系统测试 3.1 冒烟测试 3.2 回归测试 4. 验收测试 二、按照是否手工测试分类 1. 手工测试 2. 自动化测试 3. 手工测试和自动化测试的优缺点 三、按照实施组织分类 1. α测试(Alpha Testing) 2. β测试(Beta…