从0开始linux(19)——如何写一个linux环境下运行的shell程序

欢迎来到博主的专栏:从0开始Linux
博主ID:代码小豪

文章目录

    • bash
    • myshell
    • 源码

bash

什么?我写bash?bash作为一个大型的shell程序,甚至已经成为一种语言。博主当然没能力复刻。

博主这里写了一个仿bash的shell程序。主要目的是仿制bash的命令行是如何执行命令的,但是bash的功能远超于此。相当于是博主在班门弄斧了。这里先带大家了解一下bash。

bash是linux系统下的默认shell程序。比如我们登录linux系统后,显示出来的命令行,其实就是bash程序的命令行模式
在这里插入图片描述
我们在命令行中写的指令,其实都是交给bash解析的。而且我们在命令行中启动的程序,其父进程都是bash。因此,我们打开linux系统所见的那个黑乎乎的命令行,其实就是bash,换句话说,我们在使用linux的时候,其实一直都在使用bash程序,只是我们对其不了解罢了。

myshell

博主将这个自己写的shell程序,命名为myshell,既然myshell要仿bash的命令行模式,首先我们要搞清楚bash的功能有什么。

首先是命令行,我们要先提示用户,当前我们处于命令行模式,请用户输入指令。因此,我们要先写出myshell的命令行提示符。为了和bash做出区分,因此命令行设计也要和bash不同才行。
在这里插入图片描述
其具体格式为[用户名@主机名 当前工作文件]:,而这些信息都在环境变量当中,因此需要用到getenv()函数获取这些数据。

//博主在文章末尾会附上完整代码,因此这里只展示部分,以提供思路参考

std::string getusr()//获取用户名
{
    std::string name=getenv("USER");
    return name.empty()?"None":name;
}

void  PrintCommandLine()//打印命令行提示符        
{                                                 
   char buf[BASE_SIZE]={0};                       
   snprintf(buf,BASE_SIZE,"[%s@%s %s]:"\          
            ,getusr().c_str(),gethost().c_str(),getcwd().c_str());//获取这些环境变量,并且将其输出到buf中
   fputs(buf,stdout);                             
   fflush(stdout);                               
}

接下来,命令行可以接收用户输入的命令,我们将其保存在一个字符数组当中。

int main()
{
    char commandbuf[BASE_SIZE];//保存用户命令
    EnvInit();//初始化环境变量
    while(true)
    {
     	//1. 命令行提示符
        PrintCommandLine();

        //2. 获取用户命令
        if(GetUsrCommand(commandbuf)==false)
        {
            continue;
        }

		//3. 解析用户命令
        ParseUsrCommand(commandbuf);

        //4. 执行用户命令
		ExecuteUsrCommand();
    }
    return 0;
}

第三步是解析用户命令行参数,比如ls -a -l --color,我们应该将每个单独的字符串(即空格隔开的字符串),按照顺序保存在一个字符型指针数组当中。
在这里插入图片描述
第四步是执行用户命令,在前面的进程章节中,博主提到,在命令行执行的程序,实际上都是bash创建的子进程,但是bash是如何创建的呢?其实原理很简单,bash使用fork函数,创建一个子进程,接着用execvpe函数,切换到用户想要启动的进程,比如bash执行ls指令,本质上就是bash先fork出一个子进程,接着execvpe("ls",gargv,environ)的方式创建的。具体的细节大家可以去看看博主前面写的,与进程相关的文章,实际上博主写这个myshell的本质是想让大家将进程系列的知识串联起来。

void ExecuteUsrCommand()//执行用户命令
{
    pid_t id=fork();//创建一个bash子进程
    if(id==0)
    {
        execvpe(gargv[0],gargv,env);//切换到用户的命令进程
        exit(1);
    }
    else if(id>0)
    {
        int status=0;
        waitpid(id,&status,0);//回收创建的子进程,避免产生僵尸进程                                                                                                                                        
        lastcode=WEXITSTATUS(status);                            
    }                                                            
}

但是有些命令,是不能通过创建子进程的方式执行的,比如cd指令,我使用这个指令是想让bash切换工作路径,而是让子进程切换路劲,因此如果解析出用户使用cd指令,我们应该让bash自己去执行,而非创建子进程执行,这种命令,我们称其为内建命令。bash的内建命令比较多,博主只写了其中几个。

bool CheckAndExecuteBulitCommand()//判断一下用户输入的是否是内建命令
{
    if(gargc==2&&strcmp(gargv[0],"cd")==0)//cd是内建命令
    {
     	if(chdir(gargv[1])==-1)//切换myshell的当前工作路径
        {
            printf("No such file of dirtory");
            lastcode=ERROR;
            return true;
        }
		lastcode=0;
        return true;

    }
    else if(strcmp(gargv[0],"export")==0)
    {
     	addenv(gargv[1]);
        return true;
    }
    else if(strcmp(gargv[0],"env")==0)//显示myshell当前的环境变量
    {
     	for(int i=0;env[i]!=nullptr;i++)
        {
            printf("env[%d]:%s\n",i,env[i]);
        }
	return true;
    }
    else if(gargc==2&&strcmp(gargv[0],"echo")==0)
    {
     	if(strcmp(gargv[1],"$?")==0)
        {
            printf("%d\n",lastcode);
        }
		else
		{
            printf("%s\n",gargv[1]);
        }
		lastcode=0;
		return true;
    }
    else if(strcmp(gargv[0],"exit")==0)
    {
     	printf("thank you for your using myshell\n");
        exit(0);
    }

    return false;
}


到此,一个简单的shell程序myshell就已经写好了,下面是实际使用的演示。

myshell

源码

点击下面链接获取源码与程序
简单的仿bash的shell程序

或:

#include<cstdio>
#include<string>
#include<cstring>
#include<cstdlib>
#include<unistd.h>
#include<sys/wait.h>


#define BASE_SIZE 256
//命令行参数与环境变量
char *gargv[BASE_SIZE];
int gargc;
char* env[BASE_SIZE];
extern char** environ;
//最近一次进程的运行结果
int lastcode=0;
#define ERROR 1

//当前文件路径
char cwd[BASE_SIZE];
char cwdenv[BASE_SIZE];
void EnvInit()
{
    int index=0;
    while(environ[index]!=nullptr)
    {
     	env[index]=(char*)malloc(strlen(environ[index])+1);
        strcpy(env[index],environ[index]);
        index++;
    }
    env[index]=nullptr;
}

void addenv(char* item)//增加环境变量
{
    int index=0;
    while(env[index]!=nullptr)
    {
     	index++;
    }
    env[index]=(char*)malloc(strlen(item)+1);
    strcpy(env[index],item);
    env[index+1]=nullptr;
}

std::string getusr()//获取用户名
{
    std::string name=getenv("USER");
    return name.empty()?"None":name;
}

std::string gethost()//获取主机名
{
    std::string hostname=getenv("HOSTNAME");
    return hostname.empty()?"None":hostname;
}

std::string getcwd()//获取当前工作文件
{
   if( getcwd(cwd,BASE_SIZE)==nullptr)
   {
       strcpy(cwd,"Node");
   }
   snprintf(cwdenv,BASE_SIZE,"PWD=%s",cwd);
   putenv(cwdenv);
    return cwd;
}
void  PrintCommandLine()//打印命令行提示符
{
   char buf[BASE_SIZE]={0};
   snprintf(buf,BASE_SIZE,"[%s@%s %s]:"\
           ,getusr().c_str(),gethost().c_str(),getcwd().c_str());//获取这些环境变量,并且将其输出到buf中
   fputs(buf,stdout);
   fflush(stdout);
}

bool GetUsrCommand(char commandbuf[])
{
    if(fgets(commandbuf,BASE_SIZE,stdin)==nullptr)
    {
     	return false;
    }
    commandbuf[strlen(commandbuf)-1]='\0';
    return true;

}

void ParseUsrCommand(char* commandbuf)
{
    memset(gargv,0,BASE_SIZE*sizeof(commandbuf[0]));
    gargc=0;
    gargv[gargc++]=strtok(commandbuf," ");
    while(bool(gargv[gargc]=strtok(nullptr," ")))
    {
     	gargc++;
    }
}

bool CheckAndExecuteBulitCommand()//判断一下用户输入的是否是内建命令
{
    if(gargc==2&&strcmp(gargv[0],"cd")==0)//cd是内建命令
    {
     	if(chdir(gargv[1])==-1)//切换myshell的当前工作路径
        {
            printf("No such file of dirtory");
            lastcode=ERROR;
            return true;
        }
		lastcode=0;
        return true;

    }
    else if(strcmp(gargv[0],"export")==0)
    {
     	addenv(gargv[1]);
        return true;
    }
    else if(strcmp(gargv[0],"env")==0)//显示myshell当前的环境变量
    {
     	for(int i=0;env[i]!=nullptr;i++)
        {
            printf("env[%d]:%s\n",i,env[i]);
        }
		return true;
    }
    else if(gargc==2&&strcmp(gargv[0],"echo")==0)
    {
     	if(strcmp(gargv[1],"$?")==0)
        {
            printf("%d\n",lastcode);
        }
		else
		{
            printf("%s\n",gargv[1]);
    	}
		lastcode=0;
        return true;
    }
    else if(strcmp(gargv[0],"exit")==0)
    {
     	printf("thank you for your using myshell\n");
        exit(0);
    }

    return false;
}

void ExecuteUsrCommand()//执行用户命令
{
    pid_t id=fork();//创建一个bash子进程
    if(id==0)
    {
     	execvpe(gargv[0],gargv,env);//切换到用户的命令进程
        exit(1);
    }
    else if(id>0)
    {
     	int status=0;
        waitpid(id,&status,0);//回收创建的子进程,避免产生僵尸进程
        lastcode=WEXITSTATUS(status);
    }
}

void debug()
{
    int i=0;
    while(gargv[i])
    {
     	printf("%s\n",gargv[i]);
        i++;
    }
    printf("%d\n",gargc);
}

int main()
{
    char commandbuf[BASE_SIZE];
    EnvInit();//初始化环境变量
    while(true)
    {
     	//1. 命令行提示符
        PrintCommandLine();

        //2. 获取用户命令
        if(GetUsrCommand(commandbuf)==false)
        {
            continue;
        }

		//3. 解析用户命令
    	ParseUsrCommand(commandbuf);

    	//4. 执行用户命令
   	 	if(CheckAndExecuteBulitCommand())//判断一下用户输入的是否是内建命令
    	{
        	continue;
    	}
		ExecuteUsrCommand();
    }
    return 0;
}



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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Linux:文件系统基础命令扫盲

目录 查看目录下的文件 创建目录文件 删除目录文件 打印当前工作目录 切换工作目录 删除文件 复制文件或目录 移动文件或目录 创建文件 &#x1f680;主页&#xff1a;R6bandito_ ✈往期&#xff1a;《Linux与Windows文件共享》 查看目录下的文件 命令&#xff1a;ls …

2024年【流动式起重机司机】考试技巧及流动式起重机司机模拟考试题库

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 流动式起重机司机考试技巧是安全生产模拟考试一点通生成的&#xff0c;流动式起重机司机证模拟考试题库是根据流动式起重机司机最新版教材汇编出流动式起重机司机仿真模拟考试。2024年【流动式起重机司机】考试技巧及…

正确的功能可将热晶体管风速计线性化

处理传感器电路输出信号的电路或计算公式必须生成传感器响应的反函数。例如&#xff0c;如果传感器响应是对数函数&#xff0c;则线性化部分的响应必须是指数的。 这项工作首先获取传感器响应的 46 个离散点&#xff08;参见参考论文中的图 4&#xff09;。刚开始时&#xff0…

若依前后分离版集成积木报表进行token传递

若依分离板集成积木报表就不说了需要的请移步&#xff1a;若依前后分离版集成积木报表-CSDN博客 考虑到前端摸鱼不干活,所以一般都是前后端都干&#xff0c;我这里前后端都搞上&#xff0c;你们直接抄&#xff0c;抄完接着去摸鱼&#xff0c;代码不美观&#xff0c;轻喷 一、…

【景观生态学实验】实验一 ArcGIS地理数据处理及制图基础

实验目的 1.掌握ArcGIS软件基本操作&#xff1a;通过实验操作与学习&#xff0c;熟练掌握ArcGIS软件相关的基本操作&#xff0c;包括界面熟悉、工具栏使用、数据的加载和保存、基本数据处理操作等; 2.掌握如何使用ArcGIS进行影像拼接及裁剪&#xff1a;通过实验操作与学习&am…

ABAP SMARTFORMS(2)

1、表单接口 方法一&#xff1a;导入结构、内表&#xff0c;给全局定义传入结构体 方法二&#xff1a;只关联表&#xff0c;不关联结构,给全局定义传入结构体 GW_XYXX存的表头信息 GW_XYKQ存考勤信息,表中的每一行 初始化学员信息表的第一条数据作为表头 2、创建表头模板 该…

x-cmd mod | x sd - 搭配 fzf 实时预览文本替换效果,打造更直观高效的编辑体验

目录 介绍子命令使用案例 介绍 sd&#xff08;search & displace&#xff09;是一种查找和替换文本工具&#xff0c;使用常见的正则表达式语法&#xff0c;类似于 sed&#xff0c;但专注于替换操作&#xff0c;从而使用起来更直观、更易读。 该模块主要通过 fzf 以交互方式…

单片机STC8H8K64U开发板_RA6809开发板 驱动彩屏显示

单片机STC8H8K64U开发板&#xff0c;型号RT8H8K001 预留Type C接口&#xff0c;可供电SWD下载&#xff1a; RA6809开发板&#xff0c;型号RT6809CNN01 预留Type C接口供电&#xff0c;预留MCU接口、电容触摸屏接口、液晶屏接口&#xff1a; 双臂合一&#xff0c;驱动和控…

Error in cpuinfo: prctl(PR_SVE_GET_VL) failed 错误记录

今天在一台新机器上面搭建安装环境的时候出现了上面的错误&#xff0c;直观感觉是跟py-cpuinfo这个模块有关系的。 Error in cpuinfo: prctl(PR_SVE_GET_VL) failed 错误通常与 ARM 架构上的 CPU 信息库&#xff08;如 cpuinfo&#xff09;相关&#xff0c;特别是在尝试获取可扩…

web pdf 图片拖动图片合成

web pdf 图片拖动图片合成 先看效果 前端 合成后 1.原理 以前写过相关的帖子&#xff0c;使用的是 canva 但是这次换了一个思路使用的是图片 1.先把pdf转成图片 2.把pdf图片和目标图片传到浏览器 3.原理就和图片合成一样了。见上一篇帖子 4.后端也一样只不过这次是将位置和pd…

IntelliJ IDEA 常用快捷键详解与自定义修改方法

目录 前言1. IntelliJ IDEA 常用快捷键1.1 代码编辑快捷键1.2 代码导航快捷键1.3 重构快捷键1.4 调试快捷键 2. 如何修改 IntelliJ IDEA 快捷键2.1 打开快捷键设置界面2.2 查找和修改快捷键2.3 导入和导出快捷键配置 结语 前言 IntelliJ IDEA 是一款广受开发者欢迎的集成开发环…

vue2之混入(mixin)

Vue 2 的混入&#xff08;Mixin&#xff09;是一种在 Vue 组件中分发可复用功能的方式。通过混入&#xff0c;你可以将一些通用的组件选项&#xff08;如数据、方法、计算属性、生命周期钩子等&#xff09;提取到一个混入对象中&#xff0c;并在多个组件中重用这些选项&#xf…

基于Python大数据的招聘数据分析及大屏可视化系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

2024软考-《软件设计师》-易混淆知识点总结(1~6章)

一、计算机组成与体系结构 1.1、原码、反码、补码、移码的运算 原码&#xff1a;最高位表示符号位&#xff0c;其余低位表示数值的绝对值&#xff08;0表示正数&#xff0c;1表示负数&#xff09; 反码&#xff1a;正数的反码与原码相同&#xff0c;负数的反码是其绝对值按位…

基于ESP32的RGB便携式视频灯

基于ESP32的RGB便携式视频灯 拥有一套能够满足个人需求的灯光设备至关重要。市面上的RGB视频灯虽然功能强大&#xff0c;但往往价格不菲。我制作的这款灯是20W RGB便携式视频灯不仅满足了我的需求&#xff0c;而且成本仅为市售产品的三分之一。接下来&#xff0c;我将详细介绍这…

大模型开发实战1-QuickStart

0. 关于大模型和模型选择 由于OpenAI的ChatGPT流行&#xff0c;AI技术在大模型技术的赋能下高速发展&#xff0c;特别是2023年至今&#xff0c;国内的AI技术发展更是前所未有的景象&#xff0c;各大公司争相发布自己的大模型&#xff0c;包括百度文心一言&#xff0c;阿里同义…

正版CST电磁仿真软件:保障创新与合规的基石

在当今快速发展的科技时代&#xff0c;电磁仿真技术对于电子产品的设计、测试和优化至关重要。CST电磁仿真软件以其强大的功能和广泛的应用领域&#xff0c;成为众多企业和研究机构不可或缺的工具。然而&#xff0c;在选择使用CST软件时&#xff0c;确保使用正版软件不仅是对知…

【C++】红黑树的底层原理以及实现

#1024程序员节 | 征文# 个人主页&#xff1a;夜晚中的人海 文章目录 ⭐前言&#x1f686;一、红黑树的概念&#x1f3e0;二、红黑树的规则&#x1f384;三、红黑树的效率&#x1f3a1;四、红黑树的实现1. 基本框架2. 插入操作• 变色• 单旋 变色• 双旋 变色 3. 查找操作4. …

玄机平台-应急响应-webshell查杀

首先xshell连接 然后进入/var/www/html目录中&#xff0c;将文件变成压缩包 cd /var/www/html tar -czvf web.tar.gz ./* 开启一个http.server服务&#xff0c;将文件下载到本地 python3 -m http.server 放在D盾中检测 基本可以确认木马文件就是这四个 /var/www/html/shell.p…

高效实现聚水潭数据集成MySQL的技术案例

聚水潭奇门数据集成到MySQL的技术案例分享 在现代企业的数据管理中&#xff0c;如何高效、准确地实现不同系统之间的数据对接是一个关键问题。本文将聚焦于一个实际的系统对接集成案例&#xff1a;将聚水潭奇门平台的售后单数据集成到MySQL数据库中&#xff0c;具体方案名称为…