首页 > 运维 > linux运维 > 正文

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

看不見的法師
发布: 2025-06-21 09:34:21
原创
272人浏览过

一.项目简介

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

二.分析项目实现

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

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

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

三.逐步实现项目功能

1.获取命令行

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

int main(){
    while(!quit){
        // 2.交互问题,获取命令行内容
        interact(commandline, sizeof(commandline));
<pre class="brush:php;toolbar:false">    // 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");
}</p><p>void getpwd(){
// 通过getcwd系统接口获取并更新pwd
getcwd(pwd, sizeof(pwd));
}</p><p>void interact(char *cline, int size){
// 需要环境变量相关的系统调用函数来获取命令行提示信息
// 获取主机名
char hostname[64];
gethostname(hostname, sizeof(hostname));</p><pre class="brush:php;toolbar:false">// 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 <em>_argv[]){
int i = 0;
_argv[i++] = strtok(cline, DELIM);
while(_argv[i++] = strtok(NULL, DELIM));
return i-1;
}
登录后复制

3.指令的判断

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

int buildCommand(char </em>_argv[], int _argc){
// 4.指令的判断
// cd命令
if(_argc == 2 && strcmp(_argv[0], "cd") == 0){
// 更改目录
chdir(_argv[1]);
getpwd();
// 更改环境变量
sprintf(getenv("PWD"), "%s", pwd);
return 1;
}</p><pre class="brush:php;toolbar:false">// 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 </em>_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<stdio.h></p><h1>include<stdlib.h></h1><h1>include<unistd.h></h1><h1>include<string.h></h1><h1>include<assert.h></h1><h1>include<unistd.h></h1><h1>include<sys/wait.h></h1><h1>include<sys/types.h></h1><h1>define LEFT "["</h1><h1>define RIGHT "]"</h1><h1>define LABLE "$"</h1><h1>define DELIM " \t"</h1><h1>define LINE_SIZE 1024</h1><h1>define ARGC_SIZE 32</h1><h1>define EXIT_CODE 55</h1><p>int lastcode = 0;
int quit = 0;
char commandline[LINE_SIZE];
char *argv[ARGC_SIZE];
char pwd[LINE_SIZE];
// 自定义环境变量表,做成二维数组就需要维护了
char myenv[LINE_SIZE];
// 自定义本地变量表</p><p>const char* getusername(){
// 通过getenv()获取环境变量中的用户名
return getenv("USER");
}</p><p>void getpwd(){
// 通过getcwd系统接口获取并更新pwd
getcwd(pwd, sizeof(pwd));
}</p><p>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);</p><pre class="brush:php;toolbar:false">// 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命令行的博客能对大家有所帮助,欢迎大佬们留言或私信与我交流。

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

以上就是【Linux】实现一个简易的shell命令行的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习
PHP中文网抖音号
发现有趣的

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