当前位置: 首页 > article >正文

[Linux]自定义shell详解

自定义shell

  • 前言
  • 1.命令行提示符,字符串的打印
    • 1.1命令行提示符
    • 2.命令行字符串
  • 2.0对命令行字符串进行切割
  • 2.执行命令
  • 3.有趣的小问题
  • 完整代码

前言

写之前我们先看看一个完整的shell都包括了什么
在这里插入图片描述
$符号前面(包括这个符号)就是命令行提示符,后面就是命令行字符串了

1.命令行提示符,字符串的打印

1.1命令行提示符

命令行字符串的格式:

[用户名@主机名 路径]$

这三个其实就是环境变量,这就很简单了,getenv()获取环境变量

const char *UserName()
{
    char *username = getenv("USER");
    if (username)
        return username;
    else
        return "None";
}
const char *HostName()
{
    char *hostname = getenv("HOME");
    if (hostname)
        return hostname;
    else
        return "None";
}
const char *CurrentWorkDir()
{
    char *current = getenv("PWD");
    if (current)
        return current;
    else
        return "None";
}

2.命令行字符串

这里输入命令行字符串,我们不能用scanf,当我们的命令行字符串带有空格的时候scanf就无法控制了。所以这里我们用fgets()
在这里插入图片描述
文件的操作下节会说,先跟着用;注意这里fgets会有一个自带的换行符,所以我们要对他做一下去换行的操作。

int Interactive(char out[], int size)
{
    printf("[%s@%s %s]$ ", UserName(), HostName(), CurrentWorkDir());
    fgets(out, size, stdin);
    out[strlen(out) - 1] = '\0';
    return strlen(out);
}

2.0对命令行字符串进行切割

在这里插入图片描述

void Split(char in[])
{
    int i=0;
    argv[i++]=strtok(in,SEP);
    while(argv[i++]=strtok(NULL,SEP));
}

2.执行命令

上个文章提到了进程切换的接口,接下面我们就要选择一个来使用,我们有了文件名,又有argv数组,所以我们用execvp在这里插入图片描述

void Execute()
{
    pid_t id=fork();
    if(id==0)
    {
        execvp(argv[0],argv);
        exit(1);
    }
    int status=0;
    pid_t rid=waitpid(id,&status,0);
}

3.有趣的小问题

完成了主体部分我们先测试一下代码
在这里插入图片描述
大部分指令都可以完成但是cd指令无法成功运行
在这里插入图片描述
类似于cd1这种子进程无法执行的指令我们叫做内建命令,这种指令只有bash自己才可以执行。
引入函数chdir
在这里插入图片描述

int BuildinCmd()
{
    int ret=0;
    //如果是内建命令就是1执行,不是就是0继续向下执行
    if(strcmp("cd",argv[0])==0)
    {
        ret=1;
        char* target=argv[1];
        if(!target)
        {
            target=Home();
        }
        chdir(target);
    }
    return ret;
}

再次做测试,cd之后,我们的pwd确实改变了,但是命令行提示符并没有随之改变
在这里插入图片描述

再次修改
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
但是这里又出现了一个新的问题,我们输入…显示的就还是…我们想让他以绝对路径的形式让我们看。
在这里插入图片描述
在这里插入图片描述

int BuildinCmd()
{
    int ret = 0;
    // 如果是内建命令就是1执行,不是就是0继续向下执行
    if (strcmp("cd", argv[0]) == 0)
    {
        // 2. 执行
        ret = 1;
        char *target = argv[1]; // cd XXX or cd
        if (!target)
            target = Home();
        chdir(target);
        char temp[1000];
        getcwd(temp, 1000);
        snprintf(pwd, SIZE, "PWD=%s", temp);
        putenv(pwd);
    }
    return ret;
}

这次改完就符合我们的预期了。

完整代码

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

#define SIZE 1024
#define SEP " "
#define MAX_ARGC 64

char *argv[MAX_ARGC];
char pwd[SIZE];

const char *UserName()
{
    char *username = getenv("USER");
    if (username)
        return username;
    else
        return "None";
}
const char *HostName()
{
    char *hostname = getenv("HOME");
    if (hostname)
        return hostname;
    else
        return "None";
}
const char *CurrentWorkDir()
{
    char *current = getenv("PWD");
    if (current)
        return current;
    else
        return "None";
}
char *Home()
{
    return getenv("HOME");
}
// 打印命令行提示符
int Interactive(char out[], int size)
{
    printf("[%s@%s %s]$ ", UserName(), HostName(), CurrentWorkDir());
    fgets(out, size, stdin);
    out[strlen(out) - 1] = '\0';
    return strlen(out);
}

// 分割
void Split(char in[])
{
    int i = 0;
    argv[i++] = strtok(in, SEP);
    while (argv[i++] = strtok(NULL, SEP))
        ;
}
int BuildinCmd()
{
    int ret = 0;
    // 如果是内建命令就是1执行,不是就是0继续向下执行
    if (strcmp("cd", argv[0]) == 0)
    {
        // 2. 执行
        ret = 1;
        char *target = argv[1]; // cd XXX or cd
        if (!target)
            target = Home();
        chdir(target);
        char temp[1000];
        getcwd(temp, 1000);
        snprintf(pwd, SIZE, "PWD=%s", temp);
        putenv(pwd);
    }
    return ret;
}
// 执行命令
void Execute()
{
    pid_t id = fork();
    if (id == 0)
    {
        execvp(argv[0], argv);
        exit(1);
    }
    int status = 0;
    pid_t rid = waitpid(id, &status, 0);
}
int main()
{
    while (1)
    {
        char commandline[SIZE];
        // 1. 命令行提示符[yjt@hece-265342 shell]
        int n = Interactive(commandline, SIZE);
        if (n == 0)
            continue;
        // 2.切割命令行字符串
        Split(commandline);
        // 2.0执行内建命令
        n = BuildinCmd();
        if (n)
            continue;
        // 3.执行命令
        Execute();
    }
    return 0;
}

http://www.kler.cn/news/311828.html

相关文章:

  • HTML/CSS/JS学习笔记 Day4(CSS--C1 选择器声明)
  • .dav视频文件及格式转换
  • HashMap高频面试知识点
  • 【Ubuntu】ubuntu如何使用ufw(Uncomplicated Firewall)管理防火墙?一文带你学会!
  • Ubuntu-24.04中Docker-Desktop无法启动
  • 怎么操作使http变成https访问?
  • 力扣 中等 2300.咒语和药水的成功对数
  • OpenAI最新发布的o1-preview模型,和GPT-4o到底哪个更强?
  • 驱动---动态模块编译
  • win11开始按钮点不开(已解答)
  • sql中拼接操作
  • 从“治理”到“智理”,看大模型如何赋能智慧政务
  • Linux 信号的产生
  • Windows本地pycharm使用远程服务器conda虚拟环境
  • 【Android】Handler用法及原理解析
  • Rust编程的作用域与所有权
  • 面向开发者的LLM入门教程(学习笔记02):提示原则
  • 探索AI大模型:从入门到精通的学习路径
  • spring cxf 常用注解
  • 大数据时代的等保测评:数据安全与隐私保护
  • [数据集][目标检测]智慧养殖场肉鸡目标检测数据集VOC+YOLO格式3548张1类别
  • leetcode75. 颜色分类
  • 【HTML】入门教程
  • 【SpinalHDL】Scala编程之伴生对象
  • Vue 项目中引入 Axios 详解
  • 【论文阅读笔记】YOLOv10: Real-Time End-to-End Object Detection
  • 【高级编程】网络编程 基于 TCPUDP 协议的 Socket 编程
  • Remix 学习 - @remix-run/react 中的主要组件
  • 网络-内核是如何与用户进程交互
  • MySQL从入门到精通