数据结构(c语言版)(第2版)-习题答案(学生版).doc

更新时间: 试题数量: 购买人数: 提供作者:

有效期: 个月

章节介绍: 共有个章节

收藏
搜索
题库预览
将编号为0和1的两个栈存放于一个数组空间V[m]中,栈底分别处于数组的两端。当第0号栈的栈顶指针top[0]等于-1时该栈为空,当第1号栈的栈顶指针top[1]等于m时该栈为空。两个栈均从两端向中间增长。试编写双栈初始化、判断栈空、栈满、进栈和出栈等算法的函数。双栈数据结构的定义如下: typedef struct {int top[2],bot[2]; //栈顶和栈底指针 ElemType *V; //栈数组 int m; //栈最大可容纳元素个数 }DiStack [题目分析] 两栈共享向量空间,将两栈栈底设在向量两端,初始时,左栈顶指针为-1,右栈顶为m。两栈顶指针相向、迎面增长,栈顶指针相邻时为栈满。 [算法描述] (1)栈初始化 int Init() {S.top[0]=-1; S.top[1]=m; return 1;//初始化成功 } (2)入栈操作: int push(DiStack S,int i,int x) //i为栈号,i=0表示左栈,i=1为右栈,x是入栈元素。入栈成功返回1,失败返回0 {if(i<0||i>1){cout<<"栈号输入不对"<<endl;exit(0);} if(S.top[1]-S.top[0]==1){cout<<"栈已满"<<endl;return(0);} switch(i) {case 0:S.V[++S.top[0]]=x;return(1);break; case 1:S.V[--S.top[1]]=x;return(1); }//push (3)退栈操作 ElemType pop(DiStack S,int i) //退栈。i代表栈号,i=0时为左栈,i=1时为右栈。退栈成功时返回退栈元素否则返回-1 {if(i<0||i>1){cout<<"栈号输入错误"<<endl;exit(0);} switch(i) {case 0:if(S.top[0]==-1){cout<<"栈空"<<endl;return(-1); } else return(S.V[S.top[0]--]); case 1:if(S.top[1]==m){cout<<"栈空"<<endl;return(-1);} else return(S.V[S.top[1]++]); }//switch (4)判断栈空 int Empty() {return (S.top[0]==-1 && S.top[1]==m); } [算法讨论] 请注意算法中两栈入栈和退栈时的栈顶指针的计算。左栈是通常意义下的栈,而右栈入栈操作时,其栈顶指针左移(减1),退栈时,栈顶指针右移(加1)。【缺少答案,请补充】
从键盘上输入一个后缀表达式,试编写算法计算表达式的值。规定:逆波兰表达式的长度不超过一行,以$符号为输入结束,操作数之间用空格分隔,操作符只可能有+、-、*、/四种运算。例如:234 3+2*$。 [题目分析] 逆波兰表达式(即后缀表达式)求值规则如下:设立运算数栈OPND,对表达式从左到右扫描(读入),当表达式中扫描到数时,压入OPND栈。当扫描到运算符时,从OPND退出两个数,进行相应运算,结果再压入OPND栈。这个过程一直进行到读出表达式结束符$,这时OPND栈中只有一个数,就是结果。 [算法描述] float expr() {//从键盘输入逆波兰表达式,以'$'表示输入结束,本算法求逆波兰式表达式的值。 float OPND[30]; //OPND是操作数栈。 int top=0; //两栈初始化。 float num=0.0; //数字初始化。 cin>>x;//x是字符型变量。 while(x!='$') {switch(x) {case '0'<=x<='9': while(x>='0' &&x<='9' ||x=='.' ) //拼数 {if(x!='.')//处理整数 {num=num*10 + ord(x)-ord('0') ; cin>>x;} else//处理小数部分。 {scale=10.0; cin>>x; while(x>='0'&&x<='9') {num=num+(ord(x)-ord('0'))/scale; scale=scale*10; cin>>x;} } push(OPND,num); num=0.0;//数压入栈,下个数字初始化 case x==' ':break; //遇空格,继续读下一个字符。 case x=='+':push(OPND,pop(OPND)+pop(OPND));break; case x=='-':x1=pop(OPND);x2=pop(OPND);push(OPND,x2-x1);break; case x=='*':push(OPND,pop(OPND)*pop(OPND));break; case x=='/':x1=pop(OPND);x2=pop(OPND);push(OPND,x2/x1);break; default: //其它符号不作处理。 } cin>>x;//读入表达式中下一个字符。 }//结束while (x!='$') cout<<"后缀表达式的值为"<<pop(OPND); }//算法结束。 [算法讨论] 假设输入的后缀表达式是正确的,未作错误检查。算法中拼数部分是核心。若遇到大于等于'0'且小于等于'9'的字符,认为是数。这种字符的序号减去字符'0'的序号得出数。对于整数,每读入一个数字字符,前面得到的部分数要乘上10再加新读入的数得到新的部分数。当读到小数点,认为分数的整数部分已完,要接着处理小数部分。小数部分的数要除以10(或10的幂次),成为十分位,百分位,千分位等等,与前面部分数相加。在拼数过程中,若遇非数字字符,表示数已拼完,将数压入栈中,并且将变量num恢复为0,准备下一个数。这时对新读入的字符进入'+'、'-'、'*'、'/'及空格的判断,因此在结束处理数字字符的case后,不能加入break语句。【缺少答案,请补充】