【Linux后端服务器开发】Shell外壳——命令行解释器

目录

一、Shell外壳概述

二、描述Shell外壳原理的生动例子

三、C语言模拟实现Shell外壳


一、Shell外壳概述

在狭义上 , 我们称Linux操作系统的内核为 Linux

在广义上 , Linux发行版 == Linux内核 + 外壳程序

就比如市面上现在的redhat, centos, ubuntu等等我们耳熟能详的Linux发行版,事实上这些Linux发行版都是基于Linux操作系统的内核,然后对之加装了不同的Shell外壳 ,最终做出不同种类的Linux发行版。

我们作为用户,是不能直接去操作Linux内核的,为什么呢?

  1. 直接去操作Linxu内核成本是极高的,学习成本、操作成本都不低
  2. 用户直接操作Linux内核存在风险,所以Linux内核对用户设置了权限限制

所以我们有了shell外壳程序来间接帮助我们操作Linux内核。

windows操作系统的Shell外壳是一个窗口图形界面,所以我们可以通过这个窗口图形,接收来自用户的点击、拖拽等操作,从而使用windows的各个功能。

Linux操作系统的Shell外壳的名字叫bash,所以我们也可以通过bash来传达我们的操作,即用户对bash输入指令,从而使用Linux操作系统的功能。

简单的说,Shell外壳程序是对Linux内核的一层封装 , 架起了用户和Linux沟通的桥梁。

命令行解释器

在大多数Linux发行版本中,是没有图形化界面的,那么,Shell是如何帮助用户和Linux内核进行沟通的呢?答案是命令行解释器,也就是小黑框。命令行解释器就是Shell外壳在Linux系统中的具体表现。

命令行解释器上每一行都有输入指令的提示,用户输入指令和选项之后,Shell外壳接收并解析该指令,然后发送给Linux内核去处理执行,Linux内核处理之后将结果反馈给Shell外壳,Shell外壳将结果解析返还给用户。

二、描述Shell外壳原理的生动例子

从前有座山,山里有座村,村里有个老村长,而你是村长的儿子——王二狗。同时你们村还有个远近闻名的媒婆——王婆,在你们当地有非常不错的口碑,曾撮合成功了无数对男女。

你作为村长的儿子,也老大不小了,也到了该找对象的年纪。你作为一个纯情的男人,心理自然还想着你们村的如花姑娘。

但是你还是一个害羞的小男生,不方便也不敢直接去和如花姑娘直说,所以你就只得找王婆来代为传递你的信息。

这天,你找到了王婆,说你想找如花姑娘相亲,王婆说可以帮你办这件事,然后她就把你想找如花姑娘约会这件事告诉了如花姑娘。如花姑娘说不行,她说她并不认识王二狗,不想和他相亲。于是王婆就很直白的告诉你说,如花姑娘压根不想和你相亲,你还是放弃吧。

所以王婆便是沟通你和如花姑娘的桥梁,类比用户和操作系统内核之间的媒介作用。

你心想,作为一个纯情的男人,我是不会放弃的!所以这天你有找到王婆,说再帮我问问看吧,我真的很喜欢如花姑娘[大哭]。

王婆说好吧,那我再给你传达最后一次,不过不出所料,结局再次上演,如花再次拒绝,王婆又把这个残忍的事实传达给你。

这时你还是放不下,有跟王婆说能不能再再帮我问一次。王婆此时直接拒绝了,说不要再这样子了,这样如花姑娘会不堪其扰内心厌烦且痛苦的(王婆顾及如花姑娘的感受,事实上也是在保护如花姑娘,防止你亲自去找她,做出极端的事情)。

所以王婆拥有拒绝传达信息的权利,类比Shell外壳可以拒绝一些非法的请求,从而保护os。

不过故事仍然没有结束,你可是村长的儿子啊,你的一再要求,王婆肯定会考虑到村长的面子。但是王婆也得考虑到自己的口碑,不能因为这件事把招牌给砸了,毕竟还忙着给其他男女说媒呢。

所以这时王婆想到一个绝妙的对策,招一个实习生来办这件事(王婆:让我的实习生来给你办这件事吧,我溜了哦~)。

实习生办这件事,对王婆来说有两个好处,一是王婆可以跟村长交代这件事她一直在办着,只不是是她的实习生在负责,二是就算实习生办砸了这件事也没关系,反正只要不是她王婆办的,这个招牌就不会砸。

王婆可以找实习生执行难度大的任务,类比Shell外壳可以创建子进程去执行有风险的任务,从而不影响Shell外壳。

总结

Shell外壳是对Linux内核的封装,连接沟通了用户(需求)与Linux内核(执行),这降低了用户的操作学习成本。

Shell外壳可以传达用户指令,交给操作系统内核去执行,最终把执行结果反馈给用户。

同时Shell外壳也可以直接拒绝用户,从而保护操作系统内核。

Shell外壳也可以通过创建Shell外壳程序的子进程的方式,来执行有风险的指令,从而来保护bash即Shell外壳本身。

三、C语言模拟实现Shell外壳

命令行解释器

  • 输出提示
  • 获取用户输入
  • 执行命令

核心算法:字符串切割、进程替换、环境变量调用、重定向

源代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <assert.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>#define NUM 1024
#define OPT_NUM 64#define NONE_REDIR 0
#define INPUT_REDIR 1
#define OUTPUT_REDIR 2
#define APPEND_REDIR 3char commandLine[NUM];
char* myArgv[OPT_NUM];
int lastCode = 0;
int lastSig = 0;int redirType = NONE_REDIR;
char* redirFile = NULL;// 跳过空格
void skip_space(char* start)
{while (isspace(*start)){++start;}
}// 检查命令行是否是输入输出重定向
void command_check(char* commands)
{assert(commands);char* start = commands;char* end = commands + strlen(commands);while (start < end){if (*start == '>'){*start = 0;++start;if (*start == '>'){// 追加输出重定向// "ls -a -l >> test.txt"redirType = APPEND_REDIR;++start;}else{// 覆盖输出重定向// "ls -a -l > test.txt"redirType = OUTPUT_REDIR;}skip_space(start);redirFile = start;break;}else if (*start == '<'){// 输入重定向// "cat < test.txt"*start = 0;++start;skip_space(start);redirType = INPUT_REDIR;redirFile = start;break;}else {++start;}}
}int main()
{while (1){redirType = NONE_REDIR;redirFile = NULL;// 输入提示符char* pwd = getenv("PWD");char* token = strtok(pwd, "/");char* dir = token;while (token != NULL){dir = token;token = strtok(NULL, "/");}char* user = getenv("USER");if (strcmp(user, "root") == 0){printf("[%s@%s %s]# ", user, getenv("HOSTNAME"), dir);}else{printf("[%s@%s %s]$ ", user, getenv("HOSTNAME"), dir);}fflush(stdout);// 获取用户输入fgets(commandLine, sizeof(commandLine) - 1, stdin);commandLine[strlen(commandLine) - 1] = 0;command_check(commandLine);// 字符串切割myArgv[0] = strtok(commandLine, " ");int i = 1;// 添加、更改参数if (myArgv[0] != NULL && strcmp(myArgv[0], "ls") == 0){myArgv[i++] = (char*)"--color=auto";}if (myArgv[0] != NULL && strcmp(myArgv[0], "ll") == 0){myArgv[0] = (char*)"ls";myArgv[i++] = (char*)"-l";myArgv[i++] = (char*)"--color=auto";}// 存储参数while (myArgv[i++] = strtok(NULL, " ")){}// 如果是 cd 指令,无需创建子进程// 让 shell 执行对应的命令,本质是是执行系统接口    // 这种不需要子进程执行,而是让 shell 自己执行的命令,叫做内置/内建命令if (myArgv[0] != NULL && strcmp(myArgv[0], "cd") == 0){if (myArgv[1] != NULL){chdir(myArgv[1]);   // 跳转目录char tmp_path[1024] = {0};getcwd(tmp_path, sizeof(tmp_path) - 1);     // 将当前工作目录存入tmp_path中setenv("PWD", tmp_path, 1);     // 修改环境变量}else{chdir(getenv("HOME"));setenv("PWD", getenv("HOME"), 1);}continue;}if (myArgv[0] != NULL && myArgv[1] != NULL && strcmp(myArgv[0], "echo") == 0){if (strcmp(myArgv[1], "$?") == 0){// 查看上一条指令的退出码、退出信号printf("last exit code: %d, last exit sig: %d\n", lastCode, lastSig);}else{printf("%s\n", myArgv[1]);}continue;}// 执行命令pid_t id = fork();if (id == 0){// 命令由子进程执行,重定向的工作由子进程完成// 父进程给子进程提供信息重定向int fd = 0;switch (redirType){case NONE_REDIR:break;case INPUT_REDIR:fd = open(redirFile, O_RDONLY);if (fd < 0){perror("open");exit(errno);}dup2(fd, 0);break;case OUTPUT_REDIR:case APPEND_REDIR:umask(0);int flags = O_WRONLY | O_CREAT;if (redirType == APPEND_REDIR){flags |= O_APPEND;}else{flags |= O_TRUNC;}fd = open(redirFile, flags, 0666);if (fd < 0){perror("open");exit(errno);}dup2(fd, 1);break;default:printf("bug\n");break;}execvp(myArgv[0], myArgv);exit(1);}// 父进程int status = 0;pid_t ret = waitpid(id, &status, 0);assert(ret > 0);lastCode = (status >> 8) & 0xFF;lastSig = status & 0x7F;}return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

论文浅尝 | 少样本学习的语言模型的持续训练

笔记整理&#xff1a;王贵涛&#xff0c;东南大学硕士&#xff0c;研究方向为自然语言处理 链接&#xff1a;https://github.com/UIC-Liu-Lab/CPT 一、动机 克服灾难性遗忘&#xff08;CF&#xff09;是持续学习&#xff08;CL&#xff09;的一个主要目标。目前有许多方法&…

SGD原理及Pytorch实现

&#x1f38f;目录 &#x1f388;1 SGD       &#x1f384;1.1 原理       &#x1f384;1.2 构造       &#x1f384;1.3 参数详解——momentum ✨1 SGD 损失函数是用来度量模型输出和真实值的偏差&#xff0c;损失函数越小&#xff0c;说明我们的模型效…

804. n的阶乘

链接&#xff1a; https://www.acwing.com/problem/content/806/ 题目&#xff1a; 输入一个整数 nn&#xff0c;请你编写一个函数&#xff0c;int fact(int n)&#xff0c;计算并输出 nn 的阶乘。 输入格式 共一行&#xff0c;包含一个整数 nn。 输出格式 共一行&#xff0c;包…

深度学习笔记之Transformer(八)Transformer模型架构基本介绍

机器学习笔记之Transformer——Transformer模型架构基本介绍 引言回顾&#xff1a;简单理解&#xff1a; Seq2seq \text{Seq2seq} Seq2seq模型架构与自编码器自注意力机制 Transformer \text{Transformer} Transformer架构关于架构的简单认识多头注意力机制包含掩码的多头注意力…

40.RocketMQ之高频面试题大全

消息中间件如何选型 RabbitMQ erlang开发&#xff0c;对消息堆积的支持并不好&#xff0c;当大量消息积压的时候&#xff0c;会导致 RabbitMQ 的性能急剧下降。每秒钟可以处理几万到十几万条消息。 RocketMQ java开发&#xff0c;面向互联网集群化功能丰富&#xff0c;对在线业…

MySQL物理文件----日志文件(错误日志、通用查询日志、二进制日志、慢查询日志)

文章目录 MYSQL5.7/8.0支持的几种日志文件1、错误日志&#xff08;Error log&#xff09;2、一般或通用查询日志&#xff08;General query log&#xff09;3、二进制日志&#xff08;Binary log&#xff09;3、1 查看是否开启二进制日志3、2二进制日志开启3、3查看二进制文件位…

简单爬虫项目练习

爬虫项目练习 前言任务基本爬虫框架URL管理器Html 下载器HTML 解析器数据存储器爬虫调度器效果分析 前言 自学&#xff0c;参考书籍为 Python爬虫开发与项目实战 &#xff0c;具体参考了该书的第六章。过程中出现两个问题&#xff1a; 在 Pycharm 上实现时发现有些库名更改及…

自定义程序包不存在的解决方法

方案一&#xff1a; 在pom文件中加入以下代码 <plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-surefire-plugin</artifactId><version>2.4.2</version><configuration><skipTests>true</sk…

java并发编程原理-----线程

目录 上下文切换 java代码创建线程的两种方式 线程的五个状态 线程join方法 多线程之间的影响 上下文切换 CPU的每一个核心同一时刻只能执行一个线程&#xff0c;但是我们会发现电脑同一时刻现实会进行几千个线程&#xff0c;这就是cpu在快速的切换执行线程&#xff0c;由…

最早做「行业化」安全托管MSS的厂商,现在怎么样了?

科技云报道原创。 1家三甲医院&#xff0c;4个院区&#xff0c;9万台终端&#xff0c;数千台服务器&#xff0c;只配备1个安全运营人员&#xff0c;换做任何一家企事业单位都不敢想象&#xff0c;但武汉某医院却实现了7*24小时的自动化监测响应&#xff0c;所有威胁均可在1小时…

基础篇--初识STM32

初识STM32 STM32是什么 ST&#xff1a;意法半导体 M&#xff1a;MCU/MPU32:32位 ST累计推出了&#xff1a;5大类、18个系列、1000多个型号的Cortex内核微控制器 STM32芯片分类 ST中文社区网&#xff1a;https://www.stmcu.org.cn/ ST官网&#xff1a;https://www.st.com …

【从零开始学习CSS | 第一篇】选择器介绍

目录 前言&#xff1a; 选择器介绍&#xff1a; 各类选择器&#xff1a; 总结&#xff1a; 前言&#xff1a; 本文以及后续几篇文章我们将会集中介绍CSS中的常见选择器&#xff0c;选择器的出现可以让我们实现对具体的元素标签进行定制&#xff0c;因此我们要掌握好各类选择…