【Linux】编写一个简易的shell

思维导图

学习目标

       将简易的shell代码进行编写。

一、阐述shell的基本思路

       在进程程序替换中,我们可以将一个指令交给子进程,让子进程去完成这个指令。如果这个命令是一个内建命令,我们需要将这个命令交给bash进行处理。

       大致思路是:首先,我们先打印出来一行命令行,代表我们的主机名,名字和当前路径;之后捕获一行指令命令,将指令命令进行分割,存储在字符串指针数组中;然后,将这个字符串指针数组交给exec*函数进行程序替换。

二、输出一个命令行

2.1 思路和代码

           

       我们可以观察xshell中的这个命令行中的内容,我们可以仿效这个命令行中的内容打印出自己的xshell的命令行。现在我们应该思考从哪里获取这个内容呢??在环境变量中,我们可以发现有这些内容:

       我们可以使用getenv函数来分别获取USER、HOSTNAME、PWD的内容,之后使用snprintf函数将这个内容串联起来打印到一个字符串数组,以便将这个命名行打印出来。

const char* Getname()
{const char* name = getenv("USER");if(name == NULL) return "None";return name;
}const char* Gethostname()
{const char* hostname = getenv("HOSTNAME");if(hostname == NULL) return "None";return hostname;
}const char* Getpwd()
{const char* pwd = getenv("PWD");if(pwd == NULL) return "None";return pwd;
}void MakeCommendLine(char commend[], size_t size)
{const char* name = Getname();const char* hostname = Gethostname();const char* pwd = Getpwd();SkipPath(pwd);snprintf(commend, size, "[%s@%s %s]>" , name, hostname, strlen(pwd) == 1 ? "/" : pwd + 1);printf("%s", commend);fflush(stdout);
}

2.2 简要介绍一下snprintf函数

char *getenv(const char *name)

       函数的用途:该函数返回一个以 null 结尾的字符串,该字符串为被请求环境变量的值。如果该环境变量不存在,则返回 NULL。

2.3 简要介绍一下getenv函数

int snprintf(char *str, size_t size, const char *format, ...)
  1. 如果格式化后的字符串长度 < size,则将此字符串全部复制到str中,并给其后添加一个字符串结束符('\0');
  2. 如果格式化后的字符串长度 >= size,则只将其中的(size-1)个字符复制到str中,并给其后添加一个字符串结束符('\0'),返回值为欲写入的字符串长度。
  3. snprintf的返回值n,当调用失败时,n为负数,当调用成功时,n为格式化的字符串的总长度(不包括\0),当然这个字符串有可能被截断,因为buf的长度不够放下整个字符串。

三、获取用户命令字符串

       我们在输入指令命令时,会有空格,我们不能使用scanf函数,所以我们应该使用fgets函数,将指令命令进行接收。

int getecho(char* commend, size_t n)
{char* s = fgets(commend, n, stdin);if(s == NULL) return -1;commend[strlen(commend) - 1] = '\0';return strlen(commend);
}

简要介绍一下fgets函数

char *fgets(char *restrict str, int size, FILE *restrict stream)

函数的用途:fgets函数就是用来读取一行数据的,从第三个参数指定的流中读取最多第二个参数大小的字符到第一个参数指定的容器地址中。

函数的返回值:在正常情况下fgets()函数的返回值和它第一个参数相同。即读取到数据后存储的容器地址。但是如果读取出错或读取文件时文件为空,则返回一个空指针。

函数的注意事项:fgets()函数的眼里,换行符’\n’也是它要读取的一个普通字符而已。在读取键盘输入的时候会把最后输入的回车符也存进数组里面,即会把’\n’也存进数组里面,而又由于字符串本身会是以’\0’结尾的。所以在输入字符个数没有超过第二个参数指定大小之前,你输入n个字符按下回车输入,fgets()存储进第一个参数指定内存地址的是n+2个字节。最后面会多出一个’\n’和一个’\0’,而且’\n’是在’\0’的前面一个(\n\0)。其余部分请看大佬写的:fgets函数详解

四、切割命令字符串

       在获取到输入的指令字符串后,我们需要将指令进行切割。因为指令间隔是空格,我们可以使用strtok函数进行分割指令。

char* gargv[SIZE];void slashecho(char commend[], size_t n)
{gargv[0] = strtok(commend, SEP);int cnt = 1;while ((gargv[cnt++] = strtok(NULL, SEP))); // 故意写出赋值,
}

简要介绍一下strtok函数

char *strtok(char s[], const char *delim);

函数的用途:分解字符串为一组字符串。s为要分解的字符,delim为分隔符字符(如果传入字符串,则传入的字符串中每个字符均为分割符)。首次调用时,s指向要分解的字符串,之后再次调用要把s设成NULL。

函数的返回值:从s开头开始的一个个被分割的串。当s中的字符查找到末尾时,返回NULL。如果查找不到delim中的字符时,返回当前strtok的字符串的指针。所有delim中包含的字符都会被滤掉,并将被滤掉的地方设为一处分割的节点。 

五、创建子进程进行进程替换

5.1 检查指令是否为内建命令

       比较草率,直接利用if语句进行逐一的判断,如果成功,则是内建命令;如果失败,则是普通命令。如果是内建命令,我们可以重新创建一个函数来单独的进行内建命令的执行。

int ChickBuliding()
{int yes = 0;const char* entercommend = gargv[0];if(strcmp(entercommend, "cd") == 0){yes = 1;Cd();}else if(strcmp(entercommend, "echo") == 0 && strcmp(gargv[1], "$?") == 0){yes = 1;printf("%d\n", lastcode);lastcode = 0;}return yes;
}

       比如,说cd命令,我们可以利用chdir函数改变当前工作目录,getcwd函数将当前工作目录的绝对路径复制到参数buffer所指的内存空间中,参数size为buf的空间大小。在将获取到的路径写入cwd中,最后利用putenv函数将cwd写入环境变量中。

void Cd()
{const char* path = gargv[1];if(path == NULL) path = Home();chdir(path);// 刷新环境变量char temp[SIZE * 2];// 获取当前路径getcwd(temp, sizeof temp);// 将当前路径写入cwd中snprintf(cwd, sizeof cwd, "PWD=%s", temp);// 将cwd写入环境变量中putenv(cwd);
}

       还有一个echo $?命令,直接判断是否为这个命令,如果是这个命令,直接将lastcode返回,并将lastcode重新置为0。

5.2 指令是普通命令

       我们可以创建一个子进程,利用exec*函数进行程序进程替换。最后,让父进程进行等待,如果父进程等待成功,则检查退出码是否为0,如果不为0,将错误信息打印出来。

void executecommend()
{pid_t id = fork();if(id < 0) {Die();}else if(id == 0){execvp(gargv[0], gargv);exit(1);}else {int status = 0;pid_t rid = waitpid(id, &status, 0);if(rid > 0){lastcode = WEXITSTATUS(status);if(lastcode != 0) printf("%s:%s:%d\n", gargv[0], strerror(lastcode), lastcode);}}
}

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

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

相关文章

【更具吸引力的回答】Java中final、finally、finalize的区别(二)

当谈到Java中的final、finally和finalize关键字时&#xff0c;它们各自在编程中扮演着不同的角色。下面我将从您提出的几个方面来详细解释它们之间的区别&#xff1a; 1. final 概念与用途&#xff1a;final关键字在Java中用于表示不可变性。它可以修饰类、方法和变量&#x…

在CentOS 7服务器及Windows 10客户端间建立并配置NFS服务

在CentOS 7服务器及Windows 10客户端间建立并配置NFS服务 引言 网络文件系统(Network File System)&#xff0c;简称NFS&#xff0c;是一种分布式文件系统协议。它允许网络上的客户端机器像访问本地磁盘文件一样&#xff0c;通过网络访问服务器上的文件。在某些特定的业务场景中…

从drugbank提取药物对应的靶点和基因信息

DrugBank是一个综合性的在线数据库,它提供了关于药物的详细化学、药理、药代动力学和药物-药物相互作用的信息。这个数据库是一个宝贵的资源,广泛用于药物研究、药理学、毒理学、药物设计和个性化医疗等领域。 以下是DrugBank的一些主要特点和用途: 药物信息:DrugBank提供了…

HTTP/1.0、HTTP/1.1、HTTP/2.0区别

文章目录 区别HTTP/1.0HTTP/1.11. 持久连接&#xff08;长连接&#xff09;2. 管道化3. Host头字段4. 分块传输编码5. 缓存机制6. 请求方法 HTTP/2.01. 二进制分帧2. 多路复用3. 服务器推送4. 优先级设置5. 头信息压缩6. 安全性7. 流量控制 区别 特性HTTP/1.0HTTP/1.1HTTP/2.0…

【笔试训练】day23

一、打怪 思路 由于是先手攻击&#xff0c;如果一次攻击就能杀死小怪&#xff0c;那么说明可以为无限杀小怪。 再计算杀一只小怪要扣多少血就好了&#xff0c;再用总生命值去除这个扣血量&#xff0c;得到的就是最多杀死小怪的数量。注意&#xff0c;由于最后一定要活下来&am…

博客系统问题

1.数据库相关的问题&#xff0c;包括定义表的结构、创建数据库表、增删改查操作的实现&#xff1a; Flask程序中&#xff0c;使用了ORM(Object Relation Mapping, 对象关系映射)这种思想来定义实体类并据此创建数据库表。 创建&#xff1a;首先是在代码中定义python类&#xf…

React面试经验2

1.执行顺序题 onClick () > {//athis.setState({num: this.state.num 1,})console.log(1:,this.state.num);//bthis.setState({num: this.state.num 1,})console.log(2:,this.state.num);setTimeout(() > {//cthis.setState({num: this.state.num 1,});console.log(…

将矩阵按对角线排序(Lc1329)——排序

矩阵对角线 是一条从矩阵最上面行或者最左侧列中的某个元素开始的对角线&#xff0c;沿右下方向一直到矩阵末尾的元素。例如&#xff0c;矩阵 mat 有 6 行 3 列&#xff0c;从 mat[2][0] 开始的 矩阵对角线 将会经过 mat[2][0]、mat[3][1] 和 mat[4][2] 。 给你一个 m * n 的整…

DevEco:智能、灵活、实时的集成开发环境

引言 前端性能是一个老生常谈的话题了&#xff0c;它不单单是一个技术概念&#xff0c;而是用户体验中非常重要的一环。通常在一些面向用户的产品中它直接影响了用户转化率、粘性等重要指标。 那么是不是不在乎转化率的中后台产品就可以不在乎性能了&#xff1f;显然不是&…

API接口开发实现一键智能化自动抓取电商平台商品评论数据支持高并发免费接入示例

为了实现一键智能化自动抓取电商平台商品评论数据可支持高并发免费接入&#xff0c;你可以使用Python编程语言和相关库&#xff08;如requests、BeautifulSoup等&#xff09;来开发一个API接口&#xff0c;也可以使用封装好的api接口获取&#xff0c;注册一个api账号获取key和s…

超详细的胎教级Stable Diffusion使用教程(一)

这套课程分为五节课&#xff0c;会系统性的介绍sd的全部功能和实操案例&#xff0c;让你打下坚实牢靠的基础 一、为什么要学Stable Diffusion&#xff0c;它究竟有多强大&#xff1f; 二、三分钟教你装好Stable Diffusion 三、小白快速上手Stable Diffusion 四、Stable dif…

星途重启:244亿公里外的「旅行者1号」,修好了

2024年4月20日&#xff0c;旅行者1号工程团队时隔5个月&#xff0c;终于重新收到了来自47年前所发射的探测器传回的有效数据。 ▲收到数据当天&#xff0c;工程团队成员在NASA喷气动力实验室的会议室中欢呼。 01.关于旅行者1号 在当下5G和WIFI已经普及的时代&#xff0c;NASA喷…

【QT教程】QT6硬件数据库编程 QT硬件数据库

QT6硬件数据库编程 使用AI技术辅助生成 QT界面美化视频课程 QT性能优化视频课程 QT原理与源码分析视频课程 QT QML C扩展开发视频课程 免费QT视频课程 您可以看免费1000个QT技术视频 免费QT视频课程 QT统计图和QT数据可视化视频免费看 免费QT视频课程 QT性能优化视频免费看 免…

Oracle中blob和clob的区别和例子

在Oracle数据库中&#xff0c;BLOB&#xff08;Binary Large Object&#xff09;和CLOB&#xff08;Character Large Object&#xff09;是用于存储大量数据的两种大型对象&#xff08;LOB&#xff09;类型&#xff0c;但它们之间存在一些关键的区别。 数据存储方式&#xff1…

FFmpeg常用API与示例学习(二)

封装层 封装格式(container format)可以看作是编码流(音频流、视频流等)数据的一层外壳&#xff0c;将编码后的数据存储于此封装格式的文件之内。 封装又称容器&#xff0c;容器的称法更为形象&#xff0c;所谓容器&#xff0c;就是存放内容的器具&#xff0c;饮料是内容&…

vue3速览

在您的Vue.js 3应用中&#xff0c;createApp 是用于创建一个Vue应用实例的函数。您已经正确地引入了它并开始创建应用&#xff0c;但目前根组件内部是空的。下面我将为您展示一个更完整的例子&#xff0c;说明如何设置根组件的模板、数据、方法等选项&#xff0c;并挂载到DOM上…

流媒体学习之路(WebRTC)——GCC中ProbeBitrateEstimator和AcknowledgedBitrateEstimator的大作用(7)

流媒体学习之路(WebRTC)——GCC中ProbeBitrateEstimator和AcknowledgedBitrateEstimator的大作用&#xff08;7&#xff09; —— 我正在的github给大家开发一个用于做实验的项目 —— github.com/qw225967/Bifrost目标&#xff1a;可以让大家熟悉各类Qos能力、带宽估计能力&a…

Python | Leetcode Python题解之第71题简化路径

题目&#xff1a; 题解&#xff1a; class Solution:def simplifyPath(self, path: str) -> str:names path.split("/")stack list()for name in names:if name "..":if stack:stack.pop()elif name and name ! ".":stack.append(name)re…

W801学习笔记二十四:NES模拟器游戏

之前已经实现了NES模拟器玩游戏。W801学习笔记九&#xff1a;HLK-W801制作学习机/NES游戏机(模拟器) 现在要在新版本掌机中移植过来。 1、把NES文件都拷贝到SD卡中。 这回不会受内存大小限制了。我这里拷贝了4个&#xff0c;还可以拷贝更多。 2、应用初始化中&#xff0c;加载…

React 学习-7-组件API

设置状态:setState setState(object nextState[, function callback]) nextState&#xff0c;将要设置的新状态&#xff0c;该状态会和当前的state合并 callback&#xff0c;可选参数&#xff0c;回调函数。该函数会在setState设置成功&#xff0c;且组件重新渲染后调用。 替…