编写自己的shell解析器(一)一个简单的循环命令输入和历史打印

这是整个代码的第一部分。

代码很短,结构很简单。用到的数据结构就是一个循环数组来存储输入的命令。
命令历史记录最多为12个。

问题分析和解决部分没有引用完整函数实现即声明,可能会影响阅读。具体代码见本博客最下。

我的分析:

//命令输入
/*
 需要保证
 1.访问字符串中任意位置
 2.可以分割字符串 并 进行存储
 3.能够查看之前命令

 实现
 1.用char* 类型存储
 2.循环数组存储12个命令

 */

这里我简单介绍下遇到的问题和解决方法。

问题一

问题_1_1

很快代码便写好了,我的存储输入命令的字符串声明为:char *input_order,循环数组中存储命令的字符串数组声明为:char *the_order[12];
第一次编译,发现打印出的历史记录,所有的命令都是同一个。如,我第一次命令输入为:aaa,第二次为:bbb,输出的历史记录两次均为bbb。

查找原因_1_1

debug下发现,当input_order值改变时,命令 the_order的值立刻改变。
经过筛查,是为了达到封装性,把input_order的值到函数cin_INPUT_ORDER_to_HISTORYY中,赋值给the_order时,使用的是直接赋值:the_order = input_order。the_order中存储的是input_order的地址。这解释了为什么input_order变化时,the_input随之改变。

解决_1_1

于是我把赋值语句改成:strcpy(the_input,order_input)的形式,编译,运行。

问题1_2

这时。。发现竟然莫名其妙卡在order_input输入函数:scanf(“%s”,input_order)上。debug下显示不发访问input_order。加上取址运算符后,可以跳过这个步骤,但是input_order的值为空。

查找原因_1_2

经过分析,发现声明时(见最下面main函数)没有为指针变量赋初值,所以无法访问。
其实我很好奇之前是怎么成功的。。。

解决_1_2

把字符串声明为数组形式:char input_order[100],char the_order[12][100],编译运行成功。

具体输入代码见下

//命令 历史记录 循环队列
typedef struct ORDER_HISTORY
{
    int in;
    int out;
    char the_order[NUM_OF_ORDER][100];
}ORDER_HISTORY;

//将 新输入命令 加入 历史记录 循环队列
void cin_INPUT_ORDER_to_HISTORYY(ORDER_HISTORY *o1,char *s)
{
    //ORDER_HISTORY o = *(ORDER_HISTORY*)o1;
    strcpy(o1->the_order[o1->in],s);
    o1->in=(o1->in+1)%NUM_OF_ORDER;
    o1->out = o1->in;
}

问题二

问题_2_1

打印历史记录,当输入的命令总数不到12个时,会打印出空白。

查找原因_2_1

for循环打印时没有考虑这个问题

解决_2_1

我本来想是不是可以用sizeof或strlen得到the_order数组的大小,然后小于12个时,判断一下。但是最终我创建了一个变量来存储用来判断输入的命令够不够12个。

具体代码见下

//输出 命令 历史记录
void print_ORDER_HISTORY(ORDER_HISTORY o,int t)
{
    int i=0;
    int temp = 0;
    i = (o.in-1)%NUM_OF_ORDER;
    temp = t;
    if(t!=12)
    {
        while(t--)
        {
            i = (i+12)%NUM_OF_ORDER;
            printf("%s\n",o.the_order[i]);
            --i;
        }
    }
    else
    {
        for(;i!=o.out;--i)
        {
            i = (i+12)%NUM_OF_ORDER;
            printf("%s\n",o.the_order[i]);
        }
    }
}

其他相关代码

#include
#include

#include 
#include 
const int NUM_OF_ORDER = 12;

//命令 历史记录 循环队列
typedef struct ORDER_HISTORY
{
    int in;
    int out;
    char the_order[NUM_OF_ORDER][100];
}ORDER_HISTORY;


int main(int argc, const char * argv[]) {
    ORDER_HISTORY order_history; //初始化 命令历史存储
    int temp_count_order = 0;//存储数目
    order_history.in = 0;
    order_history.out = 0;
    memset(order_history.the_order, '\0', 1200);
    char input_order[100] = "exit";  //存储输入命令

    //输入指令
    printf("%s","ysh>");
    scanf("%s",input_order);
    temp_count_order++;
    cin_INPUT_ORDER_to_HISTORYY((ORDER_HISTORY*)&order_history,input_order);
    //循环输入指令
    while(strcmp(input_order, "exit"))
    {
        printf("%s","ysh>");
        scanf("%s",input_order);
        if(temp_count_order!=12) temp_count_order++;
        cin_INPUT_ORDER_to_HISTORYY((ORDER_HISTORY*)&order_history,input_order);
    }
     //print_ORDER_HISTORY(order_history,temp_count_order);

    return 0;
}

资源下载:
ysh包括 cd ls那些呀 jobs 前台后台ctrl+挂起呀 各个方向的管道呀。。之类的
点我下载 并给作者送点积分:D

你可能感兴趣的:(自己的shell程序,命令,shell,位置,代码,存储)