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

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

防止Linux rm命令误操作的多场景防护方案与实践

《防止Linuxrm命令误操作的多场景防护方案与实践》在Linux系统中,rm命令是删除文件和目录的高效工具,但一旦误操作,如执行rm-rf/或rm-rf/*,极易导致系统数据灾难,本文针对不同场景... 目录引言理解 rm 命令及误操作风险rm 命令基础常见误操作案例防护方案使用 rm编程 别名及安全删除