C语言模拟实现Liunx操作系统与用户之间的桥梁shell(代码详解)

news2025/1/12 22:54:34

什么是shell?

Shell(壳)是指命令行界面(CLI)或脚本语言,它为用户提供了与操作系统交互的方式。它是一个程序,从用户那里接收命令,并通过与操作系统内核交互来执行这些命令。Shell充当用户和操作系统之间的中介,允许用户执行各种任务,运行程序,操作文件和目录,并通过脚本实现任务自动化。

在Xshell上使用一下shell

在使用Shell时,您在Shell提示符中键入命令,Shell执行这些命令并返回输出。
下图中红圈中的字符串都是linux系统通过shell来执行的一些命令
在这里插入图片描述

模拟实现shell

思路

主要的程序逻辑就是在main()函数中。通过一个无限循环,不断等待用户输入命令并执行。

  • getUserCommand()获取用户输入的命令字符串。
  • commandSplit()将命令字符串按照分隔符分割成参数列表。
  • doBuildin()检查是否为内建命令,如果是,则执行相应的操作。
    如果不是内建命令,则调用execute()函数执行命令。
  • execute()函数使用fork()创建子进程,在子进程中调用execvp()函数执行命令,父进程等待子进程执行完毕。

这样,代码就完成了一个简单的Shell程序,能够处理用户输入的命令,并执行相应的操作。

代码实现

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>

#define NUM 1024
#define SIZE 64
#define SEP " "


char usercommand[NUM];
char cwd[1024];
char enval[100][100];//测试
int count_enval=0;
int lastcode;


const char *getUsername()
{
        const char *name = getenv("USER");
            if(name) return name;
                else return "none";

}
const char *getHostname()
{
        const char *hostname = getenv("HOSTNAME");
            if(hostname) return hostname;
                else return "none";

}
const char *getCwd()
{
        const char *cwd = getenv("PWD");
           if(cwd) return cwd;
                else return "none";

}

//打印命令行提示符,获取用户命令字符串
int getUserCommand(char *command, int num)
{
    printf("[%s@%s %s]# ", getUsername(), getHostname(), getCwd());
    char *r = fgets(command, num, stdin); // 最终你还是会输入\n
    if(r == NULL) return -1;
    // "abcd\n" "\n"
    command[strlen(command) - 1] = '\0'; // 有没有可能越界?不会
    return strlen(command);
}

//分割字符串
void commandSplit(char *in,char **out)
{
    int argc=0; 
    out[argc++]=strtok(in,SEP);
    while(out[argc++]=strtok(NULL,SEP));

    //for (int i=0;out[i];i++)
    //{
    //    printf("%d:%s\n",i,out[i]);
    //}
}


int execute(char *argv[])
{

         pid_t id=fork();

         if(id<0)
         {
             return -1;
         }
         if(id==0)
         {
             execvp(argv[0],argv);
             exit(1);
         }
         else
         {
             int status=0;
             pid_t rid=waitpid(-1,&status,0);
             if(rid>0)
             {
                 lastcode=WEXITSTATUS(status);
             }
         }
         return 0;
}

void cd(const char* path )
{
    chdir(path);
    char tmp[1024];
    getcwd(tmp,sizeof(tmp));
    sprintf(cwd,"PWD=%s",tmp);
    putenv(cwd);
    
}


//内键情况
int doBuildin(char *argv[])
{
   if(strcmp(argv[0],"cd")==0)
   {
       char * path=NULL;
       if(argv[1]==NULL)
       {
           path=".";
       }
       else  path=argv[1];
       cd(path);
       return 1;
   }
   else if(strcmp(argv[0],"exprot")==0)
   {
       if(argv==NULL)return 1;
       strcpy(enval[count_enval],argv[1]);
       putenv(enval[count_enval]);
       count_enval++;
       return 1;
   }
   else if(strcmp(argv[0],"echo")==0)
   {
       char *val =argv[1]+1;
       if(strcmp(val,"?")==0)
       {
           printf("%d\n",lastcode);
           lastcode=0;
       }
       else
       {
           printf("%s\n",getenv(val));
       }
       return 1;
   }
   return 0;
}



int main()
{
    while(1)
    {
		 char *argv[SIZE];
         //1.打印命令行提示符,获取用户命令字符串
         int n=getUserCommand(usercommand,sizeof(usercommand));
         if(n<0)continue;
         //2.分割字符串
         commandSplit(usercommand,argv);
        //3. 内键情况(cd,exprot...)
         n=doBuildin(argv);
         if(n) continue;
         //4.执行对应的命令
         execute(argv);
    }
    return 0;
}

运行结果:

在这里插入图片描述
在这里插入图片描述


代码讲解

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
这些头文件包含了在程序中使用的各种函数和类型的声明。

定义常量和全局变量:

#define NUM 1024
#define SIZE 64
#define SEP " "
char usercommand[NUM];
char cwd[1024];
char enval[100][100];//测试
int count_enval=0;
int lastcode;

stdio.h、stdlib.h、string.h、unistd.h、sys/types.h、sys/wait.h:包含了一些标准库函数和系统调用的声明。
NUM和SIZE:定义了一些常量,用于数组的大小。
SEP:定义了命令参数的分隔符。
usercommand[NUM]:存储用户输入的命令字符串。
cwd[1024]:存储当前工作目录的路径。
enval[100][100]:存储环境变量的字符串数组。
count_enval:环境变量计数器。
lastcode:上一个命令的退出状态码。


  • getUsername(), getHostname(), getCwd(): 这些函数用于获取当前用户名、主机名和工作目录。
  • getUserCommand(char *command, int num): 这个函数用于获取用户输入的命令,并返回命令的长度。
  • commandSplit(char *in, char *out[]): 这个函数将输入的命令字符串按照分隔符分割成一个个参数,存储在out数组中。
  • execute(char *argv[]): 这个函数使用fork()创建子进程,并在子进程中执行命令。
  • cd(const char *path): 这个函数用于实现cd命令,改变当前工作目录。
  • doBuildin(char *argv[]): 这个函数用于执行内建命令(如cd、export、echo)。

下面是对每一个函数的详细解释:

getUsername()
const char *getUsername()
{
    const char *name = getenv("USER");
    if (name)
        return name;
    else
        return "none";
}

该函数用于获取当前用户名。它通过调用getenv()函数获取环境变量"USER"的值,并返回该值作为用户名。如果环境变量不存在,则返回字符串"none"。


getHostname()
const char *getHostname()
{
    const char *hostname = getenv("HOSTNAME");
    if (hostname)
        return hostname;
    else
        return "none";
}

该函数用于获取当前主机名。它通过调用getenv()函数获取环境变量"HOSTNAME"的值,并返回该值作为主机名。如果环境变量不存在,则返回字符串"none"。


getCwd()
const char *getCwd()
{
    const char *cwd = getenv("PWD");
    if (cwd)
        return cwd;
    else
        return "none";
}

该函数用于获取当前工作目录。它通过调用getenv()函数获取环境变量"PWD"的值,并返回该值作为工作目录。如果环境变量不存在,则返回字符串"none"。


getUserCommand(char *command, int num)
int getUserCommand(char *command, int num)
{
    printf("[%s@%s %s]# ", getUsername(), getHostname(), getCwd());
    char *r = fgets(command, num, stdin);
    if (r == NULL)
        return -1;
    command[strlen(command) - 1] = '\0';
    return strlen(command);
}

该函数用于获取用户输入的命令。它首先打印出类似"[username@hostname cwd]#"的提示符,然后使用fgets()函数从标准输入中读取用户输入的命令字符串。读取成功后,函数会去掉字符串末尾的换行符,并返回命令的长度。如果读取失败,函数返回-1。


commandSplit(char *in, char *out[])
void commandSplit(char *in, char *out[])
{
    int argc = 0;
    out[argc++] = strtok(in, SEP);
    while (out[argc++] = strtok(NULL, SEP));
}

该函数将输入的命令字符串按照分隔符进行分割,并将分割得到的参数存储在out数组中。分割过程使用strtok()函数,第一次调用时传入原始字符串和分隔符,后续调用传入NULL和分隔符。分割完成后,out数组中存储了每个参数的地址。函数没有返回值,通过参数传递分割得到的参数列表。


execute(char *argv[])
int execute(char *argv[])
{
    pid_t id = fork();
    if (id < 0)
        return -1;
    else if (id == 0) // child
    {
        execvp(argv[0], argv);
        exit(1);
    }
    else // father
    {
        int status = 0;
        pid_t rid = waitpid(id, &status, 0);
        if (rid > 0)
        {
            lastcode = WEXITSTATUS(status);
        }
    }
    return 0;
}

该函数用于执行命令。它首先调用fork()函数创建一个子进程,然后在子进程中调用execvp()函数执行指定的命令。如果fork()失败,函数返回-1。如果是子进程,调用execvp()执行命令,并在执行完毕后调用exit(1)退出子进程。如果是父进程,使用waitpid()函数等待子进程的结束,并获取子进程的退出状态码。最后,将子进程的退出码存储在lastcode变量中。


cd(const char* path )
void cd(const char* path )
{
    chdir(path);
    char tmp[1024];
    getcwd(tmp,sizeof(tmp));
    sprintf(cwd,"PWD=%s",tmp);
    putenv(cwd);
}

调用chdir()函数改变当前工作目录为传入的路径。
使用getcwd()函数获取当前工作目录的路径,并存储在临时数组tmp中。
使用sprintf()函数将路径格式化为"PWD=路径"的形式,并存储在cwd字符串中。
使用putenv()函数将cwd设置为环境变量。


doBuildin(char *argv[])
//内键情况
int doBuildin(char *argv[])
{
   if(strcmp(argv[0],"cd")==0)
   {
       char * path=NULL;
       if(argv[1]==NULL)
       {
           path=".";
       }
       else  path=argv[1];
       cd(path);
       return 1;
   }
   else if(strcmp(argv[0],"exprot")==0)
   {
       if(argv==NULL)return 1;
       strcpy(enval[count_enval],argv[1]);
       putenv(enval[count_enval]);
       count_enval++;
       return 1;
   }
   else if(strcmp(argv[0],"echo")==0)
   {
       char *val =argv[1]+1;
       if(strcmp(val,"?")==0)
       {
           printf("%d\n",lastcode);
           lastcode=0;
       }
       else
       {
           printf("%s\n",getenv(val));
       }
       return 1;
   }
   return 0;
}

根据传入的参数数组argv[]判断是否是内键命令。
如果是cd命令,判断是否提供了路径参数,如果没有,则将路径设置为当前目录;否则,使用cd()函数切换到指定路径。
如果是export命令,判断是否提供了参数,如果没有,则返回;否则,将参数复制到enval数组中,并使用putenv()函数将其设置为环境变量,同时递增count_enval计数器。
如果是echo命令,从参数中提取变量名,并判断变量名是否为"?",如果是,则打印上一个命令的退出状态码;否则,使用getenv()函数获取指定变量的值,并打印出来。
如果是以上内键命令之一,返回1;否则,返回0。



int main()
{
    while(1)
    {
		 char *argv[SIZE];
         //1.打印命令行提示符,获取用户命令字符串
         int n=getUserCommand(usercommand,sizeof(usercommand));
         if(n<0)continue;
         //2.分割字符串
         commandSplit(usercommand,argv);
        //3. 内键情况(cd,exprot...)
         n=doBuildin(argv);
         if(n) continue;
         //4.执行对应的命令
         execute(argv);
    }
    return 0;
}

main()函数:
进入一个无限循环,用于不断接收用户的命令并执行。
定义参数数组argv[SIZE]。
调用getUserCommand()函数获取用户命令字符串。
如果获取失败,则继续下一次循环。
调用commandSplit()函数将命令字符串分割为参数数组argv[]。
调用doBuildin()函数判断是否是内建命令并执行。
如果是内键命令,则继续下一次循环。
调用execute()函数执行命令。
返回0,结束程序运行。

总结

代码实现了一个简单的交互式shell,能够解析用户输入的命令并执行相应的操作。内键命令包括cd切换目录,export设置环境变量,echo打印变量值。其他命令会通过fork()创建子进程并调用execvp()执行外部命令。

(本章完)

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

buildadmin+tp8表格操作(6)表格行中添加详情按钮并弹出对话框

在表格行中添加按钮&#xff0c; 是个基本操作 下面来看一下&#xff0c;dialog的 对话框中是怎么拿到数据的 有了这一行&#xff0c; 那么 它的子组件中&#xff0c; 都可以获取到这个对象&#xff0c;所以&#xff0c; 有弹出框的 Info 组件&#xff0c;也可以获取到 我们…

“智能与未来”2024世亚国际智能机器人展会(简称:世亚智博会)

随着科技的不断发展&#xff0c;智能机器人已经成为了当今社会的热门话题。无论是工业生产、医疗护理、家庭服务等领域&#xff0c;智能机器人都发挥着越来越重要的作用。而世亚智博会作为智能机器人领域的专业展会之一&#xff0c;旨在为全球的智能机器人产业提供一个展示、交…

【Linux】22、CPU 评价指标、性能工具、定位瓶颈、优化方法论:应用程序和系统

文章目录 一、评价 CPU 的指标1.1 CPU 使用率1.2 平均负载&#xff08;Load Average&#xff09;1.3 上下文切换1.4 CPU 缓存命中率 二、性能工具2.1 维度&#xff1a;从 CPU 性能指标出发&#xff0c;即当你查看某性能指标时&#xff0c;要清除知道哪些工具可以做到2.2 维度&a…

buildadmin+tp8表格操作(6.1)表格行中添加详情链接并弹出对话框

上一个是添加按钮&#xff0c;查看详情&#xff0c;这里我们在表格中添加一个列&#xff0c; 这个列中&#xff0c; 你也可以放按钮&#xff0c; 也就以放链接&#xff0c;点击后弹出详情对话框&#xff0c; 代码和添加按钮弹出还是有区别的&#xff0c; 包括 Info 的组件&…

JSP页面文本展示正常 但定义在java代码中的内容 输出在页面上会变成问号 问题解决

这里 我直接写在界面上的内容就是正常的 但是 java代码中定义的内容 就会变成问号 造成这个情况的原因可能是多样的 首先要确保JDK没问题 然后是 页面顶部配置 <% page language"java" contentType"text/html; charsetUTF-8" pageEncoding"UTF-…

在python中分别利用numpy,tensorflow,pytorch实现数据的增加维度(升维),减少维度(降维)

文章目录 前言一、使用numpy实现升维度&#xff0c;降维度二、使用TensorFlow实现升维度&#xff0c;降维度三、使用PyTorch实现升维度&#xff0c;降维度总结 前言 我们明确一下升维和降维的概念&#xff1a; 升维&#xff08;Dimensionality Augmentation&#xff09;&…

​vmware虚拟机ubuntu系统配置静态ip​

把虚拟机当成服务器&#xff0c;如果虚拟机的ip是一直变化的&#xff0c;每次远程连接需要都修改连接虚拟机的ip地址&#xff0c;这肯定是麻烦的。 一、设置一下本机的VMnet8的ip 配置路径&#xff1a;控制面板->所有控制面板项->网络和共享中心 二、首先设置NAT 选自…

python基础练习题库实验9

题目1 编写一个程序来生成一个平方数列表。 例如&#xff1a; 代码 def generate_square_numbers(num):square_list []for i in range(num):square_list.append(i ** 2)return square_listnum_squares int(input("How many square numbers to generate? "))sq…

矩阵代数概论

矩阵代数 共轭转置 对于矩阵 A [ a i j ] A[a_{ij}] A[aij​]&#xff0c;共轭矩阵被定义为 A ‾ [ a ‾ i j ] \overline{A}[\overline{a}_{ij}] A[aij​]&#xff0c;所以 A A A的共轭转置 A ‾ T A T ‾ \overline{A}^T\overline{A^T} ATAT&#xff0c;其中 A ‾ T \ov…

MTK Pump Express 快速充电原理分析

1 MTK PE 1.1 原理 在讲正文之前&#xff0c;我们先看一个例子。 对于一块电池&#xff0c;我们假设它的容量是6000mAh&#xff0c;并且标称电压是3.7V&#xff0c;换算成Wh(瓦时)为单位的值是22.3Wh(6000mAh*3.7V)&#xff1b;普通的充电器输出电压电流是5V2A(10W)&#xff0c…

数据中心标签的重要性

布线标签的实施是为了为用户今后的维护和管理带来最大的 便利&#xff0c;提高其管理水平和工作效率&#xff0c;减少网络配置时间&#xff0c;标签标识系统包括三个方面:标识分类及定义,标签和建立文档。 标签、标识的分类有哪些? 数据中心内的每一电缆、光缆、配线设备、端…

nginx代理本地服务请求,避免跨域;前端图片压缩并上传

痛点 有时用vscode进行一些测试 请求不同端口服务、或者其他服务接口时时&#xff0c;老是会报跨域&#xff0c;非常的烦 所有就想用 nginx 进行请求代理&#xff0c;来解决这个痛点 nginx 下载地址&#xff1a;nginx: download 下载到某一目录&#xff1a; window下nginx相关…

nvm 安装后出现的各种问题解决方法

1、nvm安装后无法安装node版本 首先需要确定删除了电脑上所有的node版本&#xff0c;如果不会卸载那么请移步到 查看 &#xff0c;我们是要通过nvm来下载node环境&#xff0c;所以之前下载的node有冲突&#xff0c;所以都要清除。 2、下载后的nvm环境&#xff0c;无法使用node、…

Apache Airflow (十一) :HiveOperator及调度HQL

&#x1f3e1; 个人主页&#xff1a;IT贫道_大数据OLAP体系技术栈,Apache Doris,Clickhouse 技术-CSDN博客 &#x1f6a9; 私聊博主&#xff1a;加入大数据技术讨论群聊&#xff0c;获取更多大数据资料。 &#x1f514; 博主个人B栈地址&#xff1a;豹哥教你大数据的个人空间-豹…

Django 入门学习总结2 创建一个投票系统

通过学习&#xff0c;我们可以实现一个简单的投票系统。这个投票系统有两部分组成。 公共部分&#xff0c;公众可以查看和进行投票。管理员可以进行增加、删除、修改投票信息。 这里投票系统Python语言版本为3.10.13&#xff0c;Django Web框架版本为4.2.7。 投票系统的实现…

量化交易:借助talib使用技术分析指标

什么是技术分析&#xff1f; 所谓股票的技术分析&#xff0c;是相对于基本面分析而言的。基本分析法着重于对一般经济情况以及各个公司的经营管理状况、行业动态等因素进行分析&#xff0c;以此来研究股票的价值&#xff0c;衡量股价的高低。而技术分析则是透过图表或技术指标…

2023-简单点-树莓派picamera2介绍和要点

picamera2 requirements简介preview windows预览窗口GtGL preview【有 x windwows情况下 硬件加速】Qt preview【软件加速】DRM/KMS preview【没有 x windwows情况下】NULL previewpreview的一些其他特征 配置camera配置的细节 捕捉图像capture images切换模式 requirements简…

Colab跑项目

这里写目录标题 Colab文件目录路径显示更改colab当前工作文件夹Colab挂载谷歌云盘colab使用命令&#xff08;从这开始看&#xff0c;前面no zuo no die)最紧要&#xff0c;首先&#xff0c;修改笔记本设置使用启用gpu![在这里插入图片描述](https://img-blog.csdnimg.cn/591a6c…

Sam Altman重回OpenAI,工牌成亮点

11月20日凌晨&#xff0c;Sam Altman在社交平台发布了一条内容“我第一次&#xff0c;也是最后一次穿这些。”他胸前挂着OpenAI的工牌&#xff0c;写的却是“客人04”。目前&#xff0c;Sam在OpenAI总部。 Sam在19日发了一条内容“我非常喜欢OpenAI团队”。结合微软等主要投资…

html-网站菜单-点击菜单展开相应的导航栏,加减号可切换

一、效果图 1.点击显示菜单栏&#xff0c;点击x号关闭&#xff1b; 2.点击一级菜单&#xff0c;展开显示二级&#xff0c;并且加号变为减号&#xff1b; 3.点击其他一级导航&#xff0c;自动收起展开的导航。 二、代码实现 <!DOCTYPE html> <html><head>&…