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

相关文章

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

Linux镜像文件制作方式

《Linux镜像文件制作方式》本文介绍了Linux镜像文件制作的过程,包括确定磁盘空间布局、制作空白镜像文件、分区与格式化、复制引导分区和其他分区... 目录1.确定磁盘空间布局2.制作空白镜像文件3.分区与格式化1) 分区2) 格式化4.复制引导分区5.复制其它分区1) 挂载2) 复制bootfs分区3)

Java数组动态扩容的实现示例

《Java数组动态扩容的实现示例》本文主要介绍了Java数组动态扩容的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1 问题2 方法3 结语1 问题实现动态的给数组添加元素效果,实现对数组扩容,原始数组使用静态分配

Python实现快速扫描目标主机的开放端口和服务

《Python实现快速扫描目标主机的开放端口和服务》这篇文章主要为大家详细介绍了如何使用Python编写一个功能强大的端口扫描器脚本,实现快速扫描目标主机的开放端口和服务,感兴趣的小伙伴可以了解下... 目录功能介绍场景应用1. 网络安全审计2. 系统管理维护3. 网络故障排查4. 合规性检查报错处理1.

Python轻松实现Word到Markdown的转换

《Python轻松实现Word到Markdown的转换》在文档管理、内容发布等场景中,将Word转换为Markdown格式是常见需求,本文将介绍如何使用FreeSpire.DocforPython实现... 目录一、工具简介二、核心转换实现1. 基础单文件转换2. 批量转换Word文件三、工具特性分析优点局

Springboot3统一返回类设计全过程(从问题到实现)

《Springboot3统一返回类设计全过程(从问题到实现)》文章介绍了如何在SpringBoot3中设计一个统一返回类,以实现前后端接口返回格式的一致性,该类包含状态码、描述信息、业务数据和时间戳,... 目录Spring Boot 3 统一返回类设计:从问题到实现一、核心需求:统一返回类要解决什么问题?

Java使用Spire.Doc for Java实现Word自动化插入图片

《Java使用Spire.DocforJava实现Word自动化插入图片》在日常工作中,Word文档是不可或缺的工具,而图片作为信息传达的重要载体,其在文档中的插入与布局显得尤为关键,下面我们就来... 目录1. Spire.Doc for Java库介绍与安装2. 使用特定的环绕方式插入图片3. 在指定位

Java使用Spire.Barcode for Java实现条形码生成与识别

《Java使用Spire.BarcodeforJava实现条形码生成与识别》在现代商业和技术领域,条形码无处不在,本教程将引导您深入了解如何在您的Java项目中利用Spire.Barcodefor... 目录1. Spire.Barcode for Java 简介与环境配置2. 使用 Spire.Barco

Java利用Spire.Doc for Java实现在模板的基础上创建Word文档

《Java利用Spire.DocforJava实现在模板的基础上创建Word文档》在日常开发中,我们经常需要根据特定数据动态生成Word文档,本文将深入探讨如何利用强大的Java库Spire.Do... 目录1. Spire.Doc for Java 库介绍与安装特点与优势Maven 依赖配置2. 通过替换