[Linux]自定义shell详解

ops/2024/9/25 0:37:41/

自定义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;elsereturn "None";
}
const char *HostName()
{char *hostname = getenv("HOME");if (hostname)return hostname;elsereturn "None";
}
const char *CurrentWorkDir()
{char *current = getenv("PWD");if (current)return current;elsereturn "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 cdif (!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 64char *argv[MAX_ARGC];
char pwd[SIZE];

const char *UserName()
{char *username = getenv("USER");if (username)return username;elsereturn "None";
}
const char *HostName()
{char *hostname = getenv("HOME");if (hostname)return hostname;elsereturn "None";
}
const char *CurrentWorkDir()
{char *current = getenv("PWD");if (current)return current;elsereturn "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 cdif (!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.ppmy.cn/ops/115533.html

相关文章

C#设计模式之访问者模式

总目录 前言 在软件构建过程中&#xff0c;由于需求的改变&#xff0c;某些类层次结构中常常需要增加新的行为&#xff0c;如果直接在基类中做这样的更改&#xff0c;将会给子类带来很繁重的变更负担&#xff0c;甚至破坏原有设计。如何在不更改类层次结构的前提下&#xff0c…

数据库1-1、1-n 、n-n关系实际场景

数据库1-1、1-n 、n-n关系实际场景 每种关系类型的 3 个不同场景案例&#xff1a; 1 对 1 关系&#xff08;One-to-One&#xff09; 用户与个人资料&#xff1a; 场景&#xff1a;每个用户有唯一的个人资料&#xff0c;每个个人资料只对应一个用户。例子&#xff1a;User 和…

深入剖析Docker容器安全:挑战与应对策略

随着容器技术的广泛应用&#xff0c;Docker已成为现代应用开发和部署的核心工具。它通过轻量级虚拟化技术实现应用的隔离与封装&#xff0c;提高了资源利用率。然而&#xff0c;随着Docker的流行&#xff0c;其安全问题也成为关注焦点。容器化技术虽然提供了良好的资源隔离&…

JUC并发编程_四大函数式接口和 Stream 流式计算

JUC并发编程_四大函数式接口和 Stream 流式计算 四大函数式接口Function 接口Predicate 接口Consumer 接口Supplier 接口 Stream 流式计算Stream 的中间操作filter&#xff1a;过滤流中的元素&#xff0c;只保留满足条件的元素map&#xff1a;对流中的每个元素应用一个函数&…

【HTTP】请求“报头”(Host、Content-Length/Content-Type、User-Agent(简称 UA))

Host 表示服务器主机的地址和端口号 URL 里面不是已经有 Host 了吗&#xff0c;为什么还要写一次&#xff1f; 这里的 Host 和 URL 中的 IP 地址、端口什么的&#xff0c;绝大部分情况下是一样的&#xff0c;少数情况下可能不同当前我们经过某个代理进行转发。过程中&#xf…

Redis 主从复制

1.主从复制的概念 主从复制&#xff0c;是指将一台Redis服务器的数据&#xff0c;复制到其他的Redis服务器。前者称为主节点(master)&#xff0c;后者称为从节点(slave),数据的复制是单向的&#xff0c;只能由主节点到从节点。 默认情况下&#xff0c;每台Redis服务器都是主节…

Rustrover2024.2 正式发布:个人非商用免费,泰裤辣

如果这个世界本身 已经足够荒唐 那究竟什么才能算是疯狂 爱情就是这样 一旦错过了 就会有另一个人代替 我们知道 jetbrains 在今年的早些时候正式为 rust 语言发布了专用的 IDE &#xff0c;也就是 rustrover。如今 rustrover 也正式跻身为 jetbrains IDE 系列的一员猛将。…

基于 UniApp 平台的学生闲置物品售卖小程序设计与实现

&#x1f497;博主介绍&#x1f497;&#xff1a;✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌ 温馨提示&#xff1a;文末有 CSDN 平台官方提供的老师 Wechat / QQ 名片 :) Java精品实战案例《700套》 2025最新毕业设计选题推荐…