[Linux]自定义shell详解

server/2024/9/24 19:14:58/

自定义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/server/121464.html

相关文章

使用四叉树碰撞的游戏 显微镜RPG

实现四叉树碰撞检测 //author bilibili 民用级脑的研发记录 // 开发环境 小熊猫c 2.25.1 raylib 版本 4.5 // 2024-7-14 // AABB 碰撞检测 在拖拽&#xff0c;绘制&#xff0c;放大缩小中 // 2024-7-20 // 直线改每帧打印一个点&#xff0c;生长的直线&#xff0c;直线炮弹 /…

js 深入理解类-class

目录 概述1. 类的定义2. 类构造函数2.1. 实例化2.1.1 实例化流程2.1.2 带参实例化2.1.3 执行构造函数返回的两种对象2.1.4 类构造函数和普通构造函数的区别 2.2 把类当成特殊函数2.2.1 辨别是不是函数&#xff0c;使用 typeof2.2.2 辨别是不是函数&#xff0c;是否有prototype2…

【计算机网络 - 基础问题】每日 3 题(十四)

✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

新手教学系列——为表格行增加倒计时列

在现代网页开发中,许多应用场景都需要显示倒计时功能,例如在线促销的限时抢购、拍卖会的剩余时间、或是订单支付的有效期。这类倒计时信息通常以动态的形式展现在表格中,随着时间的流逝,倒计时也会自动更新。这篇文章将带你一步步实现一个动态倒计时表格功能,展示如何为表…

渗透测试综合靶场 DC-2 通关详解

一、准备阶段 准备工具如Kali Linux&#xff0c;下载并设置DC-2靶场机。确保攻击机和靶机在同一网络段&#xff0c;通常设置为桥接模式或NAT模式。 1.1 靶机描述 Much like DC-1, DC-2 is another purposely built vulnerable lab for the purpose of gaining experience in …

从入门到精通:Linux 100个关键技术关键词

无论你是刚刚接触Linux的新手&#xff0c;还是希望进一步提升技能的中级用户&#xff0c;本指南都将是你不可或缺的学习资源。Linux 是一个强大而灵活的开源操作系统&#xff0c;广泛应用于服务器、嵌入式系统和个人电脑。通过掌握本指南中的100个关键技术关键词&#xff0c;你…

基于Java+SpringBoot+vue+elementUI私人健身教练预约管理系统设计实现

基于JavaSpringBootvueelementUI私人健身教练预约管理系统设计实现 &#x1f345; 作者主页 网顺技术团队 &#x1f345; 欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1f4dd; &#x1f345; 文末获取源码联系方式 &#x1f4dd; &#x1f345; 查看下方微信号获取联系方式 承接…

Redis: 特色,业务场景举例,底层原理,持续进阶等问题梳理

Redis 的特色 Redis 是目前使用非常广泛的中间件&#xff0c;在于它功能强大&#xff0c;持续改进&#xff0c;经久不衰主要体现在以下几点 1 ) 高性能: Redis 的底层是C语言编写的&#xff0c;它的定位是内存型数据库而且 Redis 的作者对操作系统也非常的精通它通讯方式采用了…