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;
}

   

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

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

相关文章

git的相关实用命令

参看文章&#xff1a;https://blog.csdn.net/qq_21688871/article/details/130158888 http://www.mobiletrain.org/about/BBS/159885.html 1、git commit后&#xff0c;但发现文件有误&#xff0c;不想push(提交到本地库&#xff0c;回退到暂存区&#xff09; git reset --sof…

windows ssh时出现Bad local forwarding specification的解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

原生video设置控制面板controls显示哪些控件

之前我们学习了如何使用原生video播放视频 今天来一个进阶版的——设置控制面板controls显示哪些控件 先看一下当我们使用原生video时&#xff0c;controls属性为true时&#xff0c;相关代码如下&#xff1a; 正常的控制面板默认显示的控件有&#xff1a;播放、时间线、音量调…

Vulhub-信息泄露

1.Jetty WEB-INF 敏感信息泄露漏洞&#xff08;CVE-2021-28164&#xff09; docker-compose up -d 启动环境&#xff0c;显示8080端口被占用 修改 docker-compose.yml 中的映射端口 curl 访问 http://192.168.48.129:8090/WEB-INF/web.xml 显示404&#xff1a; 通过 %2e 绕过…

Unity中Shader指令优化

文章目录 前言解析一下不同运算、条件、函数所需的指令数1、常数基本运算2、变量基本运算3、条件语句、循环 和 函数 前言 上一篇文章中&#xff0c;我们解析了Shader解析后的代码。我们在这篇文章中来看怎么实现Shader指令优化 Unity中Shader指令优化&#xff08;编译后指令…

2023-12-02 LeetCode每日一题(拼车)

2023-12-02每日一题 一、题目编号 1094. 拼车二、题目链接 点击跳转到题目位置 三、题目描述 车上最初有 capacity 个空座位。车 只能 向一个方向行驶&#xff08;也就是说&#xff0c;不允许掉头或改变方向&#xff09; 给定整数 capacity 和一个数组 trips , trip[i] …

Java——》线性数据结构

推荐链接&#xff1a; 总结——》【Java】 总结——》【Mysql】 总结——》【Redis】 总结——》【Kafka】 总结——》【Spring】 总结——》【SpringBoot】 总结——》【MyBatis、MyBatis-Plus】 总结——》【Linux】 总结——》【MongoD…

数据结构算法-选择排序算法

引言 说起排序算法&#xff0c;那可就多了去&#xff0c;首先了解什么叫排序 以B站为例&#xff1a; 蔡徐坤在B站很受欢迎呀&#xff0c;先来看一下综合排序 就是播放量和弹幕量&#xff0c;收藏量 一键三连 都很高这是通过一些排序算法 才能体现出综合排序 蔡徐坤鬼畜 按照播…

Redis中的数据结构

文章目录 第1关&#xff1a;Redis中的数据结构 第1关&#xff1a;Redis中的数据结构 这是上篇文章的第一关&#xff0c;只不过本篇是代码按行做的&#xff0c;方便一下大家使用。 代码如下&#xff1a; redis-cliset hello redislpush educoder-list hellorpush educoder-lis…

uniapp uview u-input在app(运行在安卓基座上)上不能动态控制type类型(显隐密码)

开发密码显隐功能时&#xff0c;在浏览器h5上功能是没问题的 <view class"login-item-input"><u-input:type"showPassWord ? password : text"style"background: #ecf0f8"placeholder"请输入密码"border"surround&quo…

mybatis数据输入-Map类型参数输入

1、建库建表 CREATE DATABASE mybatis-example;USE mybatis-example;CREATE TABLE t_emp(emp_id INT AUTO_INCREMENT,emp_name CHAR(100),emp_salary DOUBLE(10,5),PRIMARY KEY(emp_id) );INSERT INTO t_emp(emp_name,emp_salary) VALUES("tom",200.33); INSERT INTO…

【Python表白系列】玫瑰花的浪漫告白(完整代码)

文章目录 玫瑰花环境需求完整代码普通玫瑰花三维玫瑰花多彩玫瑰花系列文章玫瑰花 环境需求 python3.11.4PyCharm Community Edition 2023.2.5pyinstaller6.2.0(可选,这个库用于打包,使程序没有python环境也可以运行,如果想发给好朋友的话需要这个库哦~)【注】 python环境搭…