Linux学习——模拟实现mybash小程序

2023-12-03 10:12

本文主要是介绍Linux学习——模拟实现mybash小程序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

一,跟正宗的bash见个面

二,实现一个山寨的bash

1.提示符

2.输入命令与回显命令

 3.解析命令

4.执行命令

5.执行逻辑

三,全部代码


一,跟正宗的bash见个面

 在这篇文章中,我会写一个myshell小程序。这个小程序其实就是一个解释器。在我的机器上它长这样:

   

二,实现一个山寨的bash

1.提示符

在图:

中。这个提示符的信息可以分为四类:

1.用户名    2.主机名   3.当前地址   4.其他字符

在这个图片里:cq就是用户名  VM-8-9-centos就是主机名   mybash就是当前所在路径。

那我们该如何获取呢?两条路:1.其它字符直接打印  2.用户名等用环境变量获取。代码如下:

 #include<stdio.h>    
#include<stdlib.h>//getenv的头文件    const char* Username()//获取用户名    
{    const char* user = getenv("USER");    if(user) return user;    return "none";    
}    const char* Hostname()//获取主机名    
{    const char* host = getenv("HOSTNAME");    if(host) return host;    return "none";    
}    const char* Pwd()  //获取当前地址  
{    const char* pwd = getenv("PWD");                                                                                                                                                         if(pwd) return pwd;    return "none";    
}    int main()    
{    printf("[%s@%s %s]#\n",Username(),Hostname(),Pwd());    return 0;    
}    

效果:

可以看到我们当前的提示符显示是可以成功的。

2.输入命令与回显命令

    想到输入和显示命令时,我猜很多同学的脑子里第一个想到的便是scanf和printf。但是在这里我们是不能使用scanf的。因为我们在输入命令的时候一定会遇到输入空格的情况,如:ls -a -l命令等等。但是scanf在遇到空格的时候便会停下。所以我们不能使用scanf进行读取数据。所以我们采用gets或者fgets来读取数据。

   这两个函数介绍如下:

gets函数是将键盘上的输入读取到str缓冲区里存起来。

fgets的功能跟gets一样,这三个参数的意思如下:1.str表示存储读取到数据的地方

2.num 存储的最大数据量  。 3.stream表示从何读取(键盘读取则为stdin)。

写出代码如下:

 char buff[1024]\\一般将这个数组定义为全局的fgets(buff,sizeof(buff),stdin);                                                                                                                                                          buff[strlen(buff)-1] = '\0'; \\将回车符给吞掉  printf("%s\n",buff);  

封装函数如下:

void  getCommand()    
{    fgets(buff,sizeof(buff),stdin);    buff[strlen(buff)-1] = '\0';    
}  

效果:

 3.解析命令

  在这里解析命令的意思便是将一个长字符串以空格为分隔符分割成一个一个短的字符串。比如"ls -a -l"就应该分成"ls" "-a" "-l"。在这里我们要使用到一个字符串分割函数:

这个函数的参数:str表示要分割的字符串  delimiters表示分割符。并且要注意的是,当我的第一次分割成功以后,我后面的连续分割就可以将str用NULL表示。先在写出代码如下:

void splictCommand(char* in,char* out[]) \\注意这里的参数in是buff,out是char* argv。这两个参数都定义在全局   {    int argc = 0;    out[argc++]= strtok(in,SEP);    while(out[argc++]= strtok(NULL,SEP));   #ifdef Debug  \\用来测试  for(int i = 0;out[i];i++)    printf("%d:%s\n",i,out[i]);    #endif                                                                                                                                                                                   }   

效果:分割完成!!!

4.执行命令

在完成输入和解析命令以后我们就得来执行命令了。我们如何实现命令的执行呢?

1.创建子进程    2.使用程序替换。

在这里要了解的是,有一些命令是必须要让父进程来执行的。比如:cd export echo等。这些命令叫做内建命令。还有一些命令则不需要由父进程来来执行而是要交由子进程来执行。所以我们得创建子进程。 在执行命令的时候步骤如下:

1.先检查是否是内建命令:若是便执行并且返回一个1。若不是便返回0。

代码:

int dobuildin(char* argv[]){if(strcmp(argv[0],"cd")== 0)//cd是内建命令{char* path = NULL;if(argv[1] == NULL)  path =getHome();else  path = argv[1];cd(path);  return 1;}else if(strcmp(argv[0],"export") == 0)//export是内建命令{if(argv[1]== NULL) return 1; strcpy(enval,argv[1]);putenv(enval);return 1;}else if(strcmp(argv[0],"echo")==0)//echo是内建命令{                                                                                                                                                                                      if(argv[1] == NULL){printf("\n");}else{if(argv[1] == NULL) {printf("\n");return 1;}if(*(argv[1])=='$'&&strlen(argv[1])>1){char* val = argv[1]+1;if(strcmp(val,"?")==0){printf("%d\n",lastcode);lastcode = 0;}else printf("%s\n",getenv(val));}else{printf("%s\n",argv[1]);}return 1;}}return 0;//不是内建命令便返回0}

然后才是执行其它命令:

void excute(char* argv[]){pid_t id = fork();//创建子进程if(id == 0)//子进程执行程序替换{execvp(argv[0],argv);exit(1);//执行完便可以退出}else {int status = 0;pid_t  rid = waitpid(id,&status,0);//等待子进程if(rid>0){//等待成功lastcode = WEXITSTATUS(status);//获取最后一次的退出码}}}

执行逻辑:

 n = dobuildin(argv);//检查并执行内建命令if(n) continue;excute(argv);//子进程执行命令

这两个函数的执行顺序如上。如果内建命令执行成功在这一次便可以不再执行下面的普通命令的代码。如果不成功便可以执行下面的普通命令的代码。

5.执行逻辑

int main(){while(1){   char Usercommand[NUM];int n  =  getCommand(Usercommand,sizeof(Usercommand));//获取命令if(n<=0) continue;char* argv[SIZE];splictCommand(Usercommand,argv);//将命令打散放到数组中                                                                                                                               n = dobuildin(argv);//检查并执行内建命令if(n) continue;excute(argv);//子进程执行命令}return 0;}

三,全部代码

#include<stdio.h>
#include<stdlib.h>//getenv的头文件
#include<string.h>
#include<unistd.h>//fork的头文件
#include<sys/types.h>//要使用pid_t必须包含的头文件  
#include<wait.h>#define Debug 1
char buff[1024];
char* argv[64];
char enval[1024];//用来存储全局的环境变量
char cwd[1024];
int lastcode = 0;#define SEP " "
const char* Username()//获取用户名
{const char* user = getenv("USER");if(user) return user;return "none";
}const char* Hostname()//获取主机名
{const char* host = getenv("HOSTNAME");if(host) return host;return "none";
}const char* Pwd()
{const char* pwd = getenv("PWD");if(pwd) return pwd;return ".";
}char* getHome()
{char*  home = getenv("HOME");if(home) return home;return(char*) "none";
}int  getCommand()
{printf("[%s@%s %s]#",Username(),Hostname(),Pwd());char* str = fgets(buff,sizeof(buff),stdin);buff[strlen(buff)-1] = '\0';if(str) return strlen(str)-1;return -1;
}void splictCommand(char* in,char* out[])
{int argc = 0;out[argc++]= strtok(in,SEP);while(out[argc++]= strtok(NULL,SEP));
#ifdef Debugfor(int i = 0;out[i];i++)printf("%d:%s\n",i,out[i]);
#endif
}void cd( char* path)
{if(path == NULL){path = getHome();}int i= chdir(path);printf("%d\n",i);char temp[1024];getcwd(temp,sizeof(temp));//获取pwd并放到临时变量temp中sprintf(cwd,"PWD=%s",temp);将pwd放到全局变量cwd中putenv(cwd);//用cwd替换掉PWD内的内容实现改变PWD的目的
}int  dobuildin( char* argv[])
{if(strcmp(argv[0],"cd") == 0){char* path = argv[1];      cd(path);return 1;}else if(strcmp(argv[0],"export")== 0){char* val = argv[1];if(val == NULL) return 1;strcpy(enval,val);putenv(enval);return 1;}else if(strcmp(argv[0],"echo")== 0){if(*argv[1]=='$'&&strlen(argv[1])>1){char* val = argv[1]+1;//$?,$PATHif(strcmp(val,"?")==0){printf("%d\n",lastcode);//显示最近一次错误码lastcode = 0;return 1;}    else {printf("%s\n",getenv(val));} }else{printf("%s\n",argv[1]);}return 1;}return 0;
}void excute(char* argv[])
{pid_t id =fork();if(id == 0)//子进程 {execvp(argv[0],argv);exit(1);}else//父进程{int status = 0;pid_t rid = waitpid(id,&status,0);//等待子进程if(rid>0){lastcode = WEXITSTATUS(status); //获取退出码} } 
}int main()
{while(1){int n = getCommand();if(n<=0) continue;splictCommand(buff,argv);n =  dobuildin(argv);if(n) continue;excute(argv); }return 0;
}

   

这篇关于Linux学习——模拟实现mybash小程序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/449060

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

linux-基础知识3

打包和压缩 zip 安装zip软件包 yum -y install zip unzip 压缩打包命令: zip -q -r -d -u 压缩包文件名 目录和文件名列表 -q:不显示命令执行过程-r:递归处理,打包各级子目录和文件-u:把文件增加/替换到压缩包中-d:从压缩包中删除指定的文件 解压:unzip 压缩包名 打包文件 把压缩包从服务器下载到本地 把压缩包上传到服务器(zip

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi