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

[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;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;
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • SpringBoot 基于 Vue 的地方美食分享网站
  • 秦时明月6.2魔改版+GM工具+虚拟机一键端
  • 图片压缩格式自适应,真的很省流量!
  • 鸿蒙OpenHarmony【轻量系统内核通信机制(消息队列)】子系统开发
  • --芯片测试--
  • ARM架构下的多核处理器设计?
  • 百度移动刷下拉词工具:快速出下拉词的技术分析
  • 如何来写一份开题报告?
  • docker部署Stirling-PDF
  • 大模型-模型架构-主流架构
  • 数据结构应试-树和二叉树
  • 数据结构和算法之树形结构(1)
  • SaaS架构:流程架构分析
  • web 动画库
  • C++第八节课 日期类的补充
  • 《剑指offer》分解让复杂问题更简单
  • 【vuex入门系列02】mutation接收单个参数和多个参数
  • 【跃迁之路】【444天】程序员高效学习方法论探索系列(实验阶段201-2018.04.25)...
  • 【跃迁之路】【735天】程序员高效学习方法论探索系列(实验阶段492-2019.2.25)...
  • CNN 在图像分割中的简史:从 R-CNN 到 Mask R-CNN
  • Git同步原始仓库到Fork仓库中
  • JS函数式编程 数组部分风格 ES6版
  • Mithril.js 入门介绍
  • Nodejs和JavaWeb协助开发
  • React-Native - 收藏集 - 掘金
  • Redux 中间件分析
  • SOFAMosn配置模型
  • Theano - 导数
  • 不发不行!Netty集成文字图片聊天室外加TCP/IP软硬件通信
  • 使用API自动生成工具优化前端工作流
  • 验证码识别技术——15分钟带你突破各种复杂不定长验证码
  • ionic异常记录
  • 继 XDL 之后,阿里妈妈开源大规模分布式图表征学习框架 Euler ...
  • ​一文看懂数据清洗:缺失值、异常值和重复值的处理
  • ‌‌雅诗兰黛、‌‌兰蔻等美妆大品牌的营销策略是什么?
  • # 深度解析 Socket 与 WebSocket:原理、区别与应用
  • #QT(一种朴素的计算器实现方法)
  • #图像处理
  • (1)(1.19) TeraRanger One/EVO测距仪
  • (2021|NIPS,扩散,无条件分数估计,条件分数估计)无分类器引导扩散
  • (javascript)再说document.body.scrollTop的使用问题
  • (二十六)Java 数据结构
  • (附源码)ssm户外用品商城 毕业设计 112346
  • (每日持续更新)jdk api之StringBufferInputStream基础、应用、实战
  • (转)Mysql的优化设置
  • (转)nsfocus-绿盟科技笔试题目
  • (转载)利用webkit抓取动态网页和链接
  • .dwp和.webpart的区别
  • .NET Core中的去虚
  • .NET分布式缓存Memcached从入门到实战
  • .NET开源快速、强大、免费的电子表格组件
  • /var/lib/dpkg/lock 锁定问题
  • [ SNOI 2013 ] Quare
  • [17]JAVAEE-HTTP协议
  • [bzoj1324]Exca王者之剑_最小割