0

0

【Linux】实现一个简易的shell命令行

看不見的法師

看不見的法師

发布时间:2025-06-21 09:34:21

|

339人浏览过

|

来源于php中文网

原创

一.项目简介

本项目旨在实现一个简易的linux shell命令行。我们将通过逐步分析和实现来构建这个shell,使其能够执行基本的命令行操作。

二.分析项目实现

【Linux】实现一个简易的shell命令行

实现一个shell需要循环以下过程:

  • 获取命令行
  • 解析命令行
  • 建立一个子进程(fork)
  • 替换子进程(execvp)
  • 父进程等待子进程退出(wait)

三.逐步实现项目功能

1.获取命令行

我们将获取命令行设计为一个循环,除非用户主动退出,否则一直保持命令行接收指令的状态。具体实现逻辑如下:

int main(){
    while(!quit){
        // 2.交互问题,获取命令行内容
        interact(commandline, sizeof(commandline));
    // 3.分割命令字符串strtok(),解析命令行
    int argc = splitstring(commandline, argv);
    if(argc == 0) continue;

    // 4.指令的判断
    int n = buildCommand(argv, argc);

    // 5.普通命令的执行
    if(!n) NormalExcute(argv);
}
return 0;

}

具体的获取命令行逻辑如下函数:

const char* getusername(){
// 通过getenv()获取环境变量中的用户名
return getenv("USER");
}

void getpwd(){ // 通过getcwd系统接口获取并更新pwd getcwd(pwd, sizeof(pwd)); }

void interact(char *cline, int size){ // 需要环境变量相关的系统调用函数来获取命令行提示信息 // 获取主机名 char hostname[64]; gethostname(hostname, sizeof(hostname));

// 1.打印bash命令行前面的提示信息
getpwd();
printf(LEFT"%s@%s %s"RIGHT""LABLE" ", getusername(), hostname, pwd);

// 2.接收用户输入信息
fgets(cline, size, stdin);
assert(cline != NULL);
(void)cline; // 防止编译器报错定义而未使用的变量(假装用一下)
cline[strlen(cline)-1] = '\0';

}

2.解析命令行

解析命令行主要是将获取到的字符串按空格切分开来放入一个新数组中。我们使用strtok()来完成这个工作,具体实现代码如下:

int splitstring(char cline[], char _argv[]){
int i = 0;
_argv[i++] = strtok(cline, DELIM);
while(_argv[i++] = strtok(NULL, DELIM));
return i-1;
}

3.指令的判断

虽然我们可以借助fork()创建子进程来实现诸多普通命令,但对于很多内建命令来说,创建子进程执行命令的结果并不会影响父进程,这会导致父进程命令无效。因此对于内建命令我们要先判断,再让父进程自主完成这些内建命令,代码如下:

易想商务网
易想商务网

YxB2B商务网是易想网络旗下的门户型B2B行业网站系统,采用先进的标签技术和静态生成技术,通过网站后台管理轻松实现网站前台多种风格和会员网站多风格,让每一个只要懂得简单网页制作常识的网友,轻松制作出精美专业的的行业商务网站系统。系统高速、稳定、安全,完全仿阿里巴巴功能设计,有供应信息、求购信息、产品库、公司库、专项商机、行业信息、展会服务、人才市场、会员助手、网商博客、商友论坛、全方位搜索等栏目

下载
int buildCommand(char _argv[], int _argc){
// 4.指令的判断
// cd命令
if(_argc == 2 && strcmp(_argv[0], "cd") == 0){
// 更改目录
chdir(_argv[1]);
getpwd();
// 更改环境变量
sprintf(getenv("PWD"), "%s", pwd);
return 1;
}

// export命令
else if(_argc == 2 && strcmp(_argv[0], "export") == 0){
    // 因为_argv一直被我们用来存新的指令,环境变量会因此被覆盖
    // 所以需要一个固定的存环境变量的地方来保存环境变量
    strcpy(myenv, _argv[1]);
    putenv(myenv);
    return 1;
}

// echo命令
else if(_argc == 2 && strcmp(_argv[0], "echo") == 0){
    if(strcmp(_argv[1], "$?") == 0){
        printf("%d\n", lastcode);
        lastcode = 0;
    }
    else if(*_argv[1] == '$'){
        char *val = getenv(_argv[1]+1);
        if(val) printf("%s\n", val);
    }
    else {
        printf("%s\n", _argv[1]);
    }
    return 1;
}

// ls命令
if(strcmp(_argv[0], "ls") == 0){
    _argv[_argc++] = "--color";
    _argv[_argc] = NULL;
}
return 0;

}

4.普通命令的执行

普通命令的执行不会影响父进程,因此我们可以使用fork()创建子进程,然后使用exec系列进程替换函数来完成相关操作,代码如下:

void NormalExcute(char _argv[]){
// 5.普通命令的执行
pid_t id = fork();
if(id < 0){
perror("fork");
exit(1);
}
else if(id == 0){
// 子进程
execvp(_argv[0], _argv);
perror("execvp");
exit(1);
}
else{
// 父进程
int status;
waitpid(id, &status, 0);
if(WIFEXITED(status)){
lastcode = WEXITSTATUS(status);
}
}
}

四.完整项目代码

完整项目代码如下:

#include

include

include

include

include

include

include

include

define LEFT "["

define RIGHT "]"

define LABLE "$"

define DELIM " \t"

define LINE_SIZE 1024

define ARGC_SIZE 32

define EXIT_CODE 55

int lastcode = 0; int quit = 0; char commandline[LINE_SIZE]; char *argv[ARGC_SIZE]; char pwd[LINE_SIZE]; // 自定义环境变量表,做成二维数组就需要维护了 char myenv[LINE_SIZE]; // 自定义本地变量表

const char* getusername(){ // 通过getenv()获取环境变量中的用户名 return getenv("USER"); }

void getpwd(){ // 通过getcwd系统接口获取并更新pwd getcwd(pwd, sizeof(pwd)); }

void interact(char *cline, int size){ // 1.打印bash命令行前面的提示信息 // 需要环境变量相关的系统调用函数来获取命令行提示信息 char hostname[64]; gethostname(hostname, sizeof(hostname)); getpwd(); printf(LEFT"%s@%s %s"RIGHT""LABLE" ", getusername(), hostname, pwd);

// 2.接收用户输入信息
fgets(cline, size, stdin);
assert(cline != NULL);
(void)cline; // 防止编译器报错定义而未使用的变量(假装用一下)
cline[strlen(cline)-1] = '\0';

}

int splitstring(char cline[], char *_argv[]){ int i = 0; _argv[i++] = strtok(cline, DELIM); while(_argv[i++] = strtok(NULL, DELIM)); return i-1; }

void NormalExcute(char *_argv[]){ // 5.普通命令的执行 pid_t id = fork(); if(id

结语

希望这篇关于在Linux中实现一个简易的shell命令行的博客能对大家有所帮助,欢迎大佬们留言或私信与我交流。

学海漫浩浩,我亦苦作舟!关注我,大家一起学习,一起进步!

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

232

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

437

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

755

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

91

2023.09.25

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

258

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

209

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1468

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

620

2023.11.24

excel表格操作技巧大全 表格制作excel教程
excel表格操作技巧大全 表格制作excel教程

Excel表格操作的核心技巧在于 熟练使用快捷键、数据处理函数及视图工具,如Ctrl+C/V(复制粘贴)、Alt+=(自动求和)、条件格式、数据验证及数据透视表。掌握这些可大幅提升数据分析与办公效率,实现快速录入、查找、筛选和汇总。

0

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
尚观shell视频教程
尚观shell视频教程

共8课时 | 1.7万人学习

Linux开发篇视频教程
Linux开发篇视频教程

共18课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号