【Linux】实现一个简单的shell

1.shell原理

shell就是运用程序替换的原理进行实现的。
何为程序替换?
假如操作系统正在执行某一个程序,然后我们利用程序替换函 数指定一个新的程序,让操作系统去执行我们新指定的程序。也就是这样一种情形下,我们fork一个进程,如果fork成功,子进程会和父进程执行相同的代码,而我们创建子进程是希望子进程执行指定的操作,所以需要执行exec族函数。
何为程序替换函数?
程序替换函数是一族函数,可以通过man命令进行查看。
这里写图片描述


2.模拟实现一个简单的shell

实现步骤:
1)采用read函数读取标准输入写到显示器上的信息,并且返回读取到的字符个数,如果返回值大于0,说明有读取到字符,就将读到的字符串的最后加个字符串的结束符\0;否则,结束此次循环。
2)将读取到的字符串按照空格分成多个字符串,放进指针数组argv中,并在指针数组的最后加一个NULL。
3)创建一个子进程,父进程等待子进程执行完程序,子进程执行程序替换函数,关于程序替换函数的选择:我们选择的是execvp函数(因为我们已经知道要执行的程序的文件名,参数也已经全部存储在argv指针数组中)。
获得登录信息的关键函数:
1)getpwuid()
这里写图片描述
2)gethostname()
这里写图片描述
3)getcwd()函数
这里写图片描述
shell的基本功能:
1)自动获取用户名,主机名,当前目录;
2)可以实现连续按回车的情况;
3)按错命令可以有删除的功能(就是)。


3.代码实现:

//MyShell.c

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<ctype.h>
#include<pwd.h>
#include<string.h>
void GetLoginName()
{
    struct passwd* pass;
    pass = getpwuid(getuid());
    printf("[%s@",pass->pw_name);
}
void GetHostName()
{
    char name[128];
    gethostname(name,sizeof(name)-1);
    printf("%s",name);
}
void GetDir()
{
    char pwd[128];
    getcwd(pwd,sizeof(pwd)-1);
    int len = strlen(pwd);
    char* p = pwd+len-1;
    while(*p != '/' && len--)
    {
        p--;
    }
    p++;
    printf(" %s]@",p);
}
int main()
{
    while(1)
    {
        GetLoginName();
        GetHostName();
        GetDir();
        fflush(stdout);
        //读取字符串
        char buf[1024];
        int s = read(0,buf,1024);
        if(s > 0)//有读取到字符
        {
            int i = 0;
            for( i = 0; i < s; ++i)
            {
                if(buf[i] == '\b' && i >= 1)
                {
                //    printf("debug:%d\n",i);
                    int j = 0;
                    for( j = i+1; j < s; ++j)
                    {
                        buf[j-2] = buf[j];
                    }
                    buf[s-2] = 0;
                    i-=1;
                }
                else if(buf[i] == '\b' && i == 0)
                {
                //    printf("debug:%d\n",i);
                    int j = 0;
                    for( j = 1; j < s; ++j)
                    {
                        buf[j-1] = buf[j];
                    }
                    buf[s-1] = 0;
                //    i-=1;
                }
                else
                {
                    continue;
                }
            }
            buf[s] = 0;
        }
        else
        {
            continue;
        }
        //将读取到的字符串分成多个字符串
        char* start = buf;
        int i =1;
        char* MyArgv[10] = {0};
        MyArgv[0] = start;
        while(*start)
        {
            if(isspace(*start))
            {
                *start = 0;
                start++;
                MyArgv[i++] = start;
            }
            else
            {
                ++start;
            }
        }
        MyArgv[i-1] = NULL;

        //打印一下字符串信息
        int m = 0;
        for(m = 0; m <i-1;++m)
        {
            printf("debug:%s\n",MyArgv[m]);
        }
        //fork新的进程
        int  id = fork();
        if(id == 0)
        {
            //child,执行替换操作
            execvp(MyArgv[0],MyArgv);
            perror("error");
            exit(1);
        }
        else
        {
            printf("father\n");
            wait(NULL);
        }
    }
    return 0;
}

//Makefile

MyShell:MyShell.c
    gcc -o MyShell MyShell.c
.PHONY:clean
clean:
    rm -f MyShell

4.代码运行结果展示:

这里写图片描述


这里写图片描述


这里写图片描述

  • 16
    点赞
  • 117
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
好的,我可以为您介绍一下如何实现一个简单Shell程序Shell是一种命令行解释器,它能够解释用户输入的命令并执行相应的操作。Shell程序通常由用户输入命令,Shell解释命令并执行相应的操作,然后Shell等待用户下一个命令的输入。 下面是一个简单Shell程序实现,主要分为以下几个步骤: 1. 读取用户输入 使用C语言中的fgets函数从标准输入读取用户输入的命令。 2. 解析用户输入 将用户输入的命令按照空格分割成多个部分,如命令名和参数等。 3. 执行用户命令 根据用户输入的命令,使用execvp函数执行相应的系统命令,例如ls、cd等。 4. 循环执行 使用while循环不断地读取用户输入并执行相应的操作,直到用户输入exit命令退出程序。 下面是一个简单Shell程序的代码示例: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <sys/wait.h> #include <string.h> #define MAX_COMMAND_LENGTH 1024 int main() { char command[MAX_COMMAND_LENGTH]; char* args[MAX_COMMAND_LENGTH/2]; while (1) { printf("myshell> "); fflush(stdout); // 读取用户输入 fgets(command, MAX_COMMAND_LENGTH, stdin); // 去掉换行符 command[strlen(command)-1] = '\0'; // 解析用户输入 char* token = strtok(command, " "); int i = 0; while (token != NULL) { args[i++] = token; token = strtok(NULL, " "); } args[i] = NULL; // 执行用户命令 pid_t pid = fork(); if (pid == -1) { printf("fork failed\n"); } else if (pid == 0) { execvp(args[0], args); exit(0); } else { wait(NULL); } // 处理exit命令 if (strcmp(args[0], "exit") == 0) { break; } } return 0; } ``` 上面的代码中使用了fork函数创建一个新的子进程,并使用execvp函数执行用户命令。在子进程中调用execvp函数后,进程的内存空间将被覆盖为新程序的内存空间。因此,在子进程中调用exit函数退出程序。 在主进程中,使用wait函数等待子进程执行完毕。wait函数将暂停当前进程的执行,直到其子进程结束后恢复执行。 以上就是一个简单Shell程序实现。当然,实际上Shell程序实现要比这个复杂得多,因为需要考虑很多其他的因素,例如I/O重定向、管道、变量替换等等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值