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

相关文章

Python实现自动化Word文档样式复制与内容生成

《Python实现自动化Word文档样式复制与内容生成》在办公自动化领域,高效处理Word文档的样式和内容复制是一个常见需求,本文将展示如何利用Python的python-docx库实现... 目录一、为什么需要自动化 Word 文档处理二、核心功能实现:样式与表格的深度复制1. 表格复制(含样式与内容)2

python获取cmd环境变量值的实现代码

《python获取cmd环境变量值的实现代码》:本文主要介绍在Python中获取命令行(cmd)环境变量的值,可以使用标准库中的os模块,需要的朋友可以参考下... 前言全局说明在执行py过程中,总要使用到系统环境变量一、说明1.1 环境:Windows 11 家庭版 24H2 26100.4061

Python中bisect_left 函数实现高效插入与有序列表管理

《Python中bisect_left函数实现高效插入与有序列表管理》Python的bisect_left函数通过二分查找高效定位有序列表插入位置,与bisect_right的区别在于处理重复元素时... 目录一、bisect_left 基本介绍1.1 函数定义1.2 核心功能二、bisect_left 与

VSCode设置python SDK路径的实现步骤

《VSCode设置pythonSDK路径的实现步骤》本文主要介绍了VSCode设置pythonSDK路径的实现步骤,包括命令面板切换、settings.json配置、环境变量及虚拟环境处理,具有一定... 目录一、通过命令面板快速切换(推荐方法)二、通过 settings.json 配置(项目级/全局)三、

pandas实现数据concat拼接的示例代码

《pandas实现数据concat拼接的示例代码》pandas.concat用于合并DataFrame或Series,本文主要介绍了pandas实现数据concat拼接的示例代码,具有一定的参考价值,... 目录语法示例:使用pandas.concat合并数据默认的concat:参数axis=0,join=

Android学习总结之Java和kotlin区别超详细分析

《Android学习总结之Java和kotlin区别超详细分析》Java和Kotlin都是用于Android开发的编程语言,它们各自具有独特的特点和优势,:本文主要介绍Android学习总结之Ja... 目录一、空安全机制真题 1:Kotlin 如何解决 Java 的 NullPointerExceptio

Python程序打包exe,单文件和多文件方式

《Python程序打包exe,单文件和多文件方式》:本文主要介绍Python程序打包exe,单文件和多文件方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python 脚本打成exe文件安装Pyinstaller准备一个ico图标打包方式一(适用于文件较少的程

java中BigDecimal里面的subtract函数介绍及实现方法

《java中BigDecimal里面的subtract函数介绍及实现方法》在Java中实现减法操作需要根据数据类型选择不同方法,主要分为数值型减法和字符串减法两种场景,本文给大家介绍java中BigD... 目录Java中BigDecimal里面的subtract函数的意思?一、数值型减法(高精度计算)1.

C#代码实现解析WTGPS和BD数据

《C#代码实现解析WTGPS和BD数据》在现代的导航与定位应用中,准确解析GPS和北斗(BD)等卫星定位数据至关重要,本文将使用C#语言实现解析WTGPS和BD数据,需要的可以了解下... 目录一、代码结构概览1. 核心解析方法2. 位置信息解析3. 经纬度转换方法4. 日期和时间戳解析5. 辅助方法二、L

使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)

《使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)》字体设计和矢量图形处理是编程中一个有趣且实用的领域,通过Python的matplotlib库,我们可以轻松将字体轮廓... 目录背景知识字体轮廓的表示实现步骤1. 安装依赖库2. 准备数据3. 解析路径指令4. 绘制图形关键