数据结构习题解析第4章Word文档格式.docx
《数据结构习题解析第4章Word文档格式.docx》由会员分享,可在线阅读,更多相关《数据结构习题解析第4章Word文档格式.docx(38页珍藏版)》请在冰豆网上搜索。
![数据结构习题解析第4章Word文档格式.docx](https://file1.bdocx.com/fileroot1/2023-1/28/15885b87-f909-4013-88ce-5fcc9ee9acb3/15885b87-f909-4013-88ce-5fcc9ee9acb31.gif)
优先级队列的插入与删除算法
三、教材中习题的解析
4-1改写顺序栈的进栈成员函数Push(x),要求当栈满时执行一个stackFull()操作进行栈满处理。
其功能是:
动态创建一个比原来的栈数组大二倍的新数组,代替原来的栈数组,原来栈数组中的元素占据新数组的前MaxSize位置。
【解答】
template<
classType>
voidstack<
Type>
:
:
push(constType&
item){
if(isFull())stackFull();
//栈满,做溢出处理
elements[++top]=item;
//进栈
}
template<
voidstack<
stackFull(){
Type*temp=newType[3*maxSize];
//创建体积大二倍的数组
for(inti=0;
i<
=top;
i++)temp[i]=elements[i];
//传送原数组的数据
delete[]elements;
//删去原数组
maxSize*=3;
//数组最大体积增长二倍
elements=temp;
//新数组成为栈的数组空间
4-2铁路进行列车调度时,常把站台设计成栈式结构的站台,如右图所示。
试问:
(1)设有编号为1,2,3,4,5,6的六辆列车,顺序开入栈式结构的站台,则可能的出栈序列有多少种?
(2)若进站的六辆列车顺序如上所述,那么是否能够得到435612,325641,154623和135426的出站序列,如果不能,说明为什么不能;
如果能,说明如何得到(即写出"
进栈"
或"
出栈"
的序列)。
(1)可能的不同出栈序列有种。
(2)不能得到435612和154623这样的出栈序列。
因为若在4,3,5,6之后再将1,2出栈,则1,2必须一直在栈中,此时1先进栈,2后进栈,2应压在1上面,不可能1先于2出栈。
154623也是这种情况。
出栈序列325641和135426可以得到。
3
5
6
2
4
4
1
33232325325325632564325641
1113135********213542135426
4-3试证明:
若借助栈可由输入序列1,2,3,…,n得到一个输出序列p1,p2,p3,…,pn(它是输入序列的某一种排列),则在输出序列中不可能出现以下情况,即存在i<
j<
k,使得pj<
pk<
pi。
(提示:
用反证法)
充分性:
由j<
k,pj<
pk,则pj必须在pk入栈之前就出栈;
而i<
j,pj<
pi,则意味着pi必须先于pj进栈且pj必须先于pi出栈;
此外,i<
k,则表明pk必须在pi之后出栈,这与pj<
pi相矛盾(因为这意味着pj必须在pk之前和pi之后离开,但pi又出现在pk之后)。
下面详细解释一下。
借助栈由输入序列1,2,3,…,n,可得到输出序列p1,p2,p3,…,pn,如果存在下标i,j,k,满足i<
k,那么在输出序列中,可能出现如下5种情况:
①i进栈,i出栈,j进栈,j出栈,k进栈,k出栈。
此时具有最小值的排在最前面pi位置,具有中间值的排在其后pj位置,具有最大值的排在pk位置,有pi<
pj<
pk,不可能出现pj<
pi的情形;
②i进栈,i出栈,j进栈,k进栈,k出栈,j出栈。
此时具有最小值的排在最前面pi位置,具有最大值的排在pj位置,具有中间值的排在最后pk位置,有pi<
pj,不可能出现pj<
③i进栈,j进栈,j出栈,i出栈,k进栈,k出栈。
此时具有中间值的排在最前面pi位置,具有最小值的排在其后pj位置,有pj<
pi<
④i进栈,j进栈,j出栈,k进栈,k出栈,i出栈。
此时具有中间值的排在最前面pi位置,具有最大值的排在其后pj位置,具有最小值的排在pk位置,有pk<
pj,也不可能出现pj<
⑤i进栈,j进栈,k进栈,k出栈,j出栈,i出栈。
此时具有最大值的排在最前面pi位置,具有中间值的排在其后pj位置,具有最小值的排在pk位置,有pk<
pi,也不可能出现pj<
4-4将编号为0和1的两个栈存放于一个数组空间V[m]中,栈底分别处于数组的两端。
当第0号栈的栈顶指针top[0]等于-1时该栈为空,当第1号栈的栈顶指针top[1]等于m时该栈为空。
两个栈均从两端向中间增长。
当向第0号栈插入一个新元素时,使top[0]增1得到新的栈顶位置,当向第1号栈插入一个新元素时,使top[1]减1得到新的栈顶位置。
当top[0]+1==top[1]时或top[0]==top[1]-1时,栈空间满,此时不能再向任一栈加入新的元素。
试定义这种双栈(DoubleStack)结构的类定义,并实现判栈空、判栈满、插入、删除算法。
双栈的类定义如下:
#include<
assert.h>
template<
classDblStack{//双栈的类定义
private:
inttop[2],bot[2];
//双栈的栈顶指针和栈底指针
Type*elements;
//栈数组
intm;
//栈最大可容纳元素个数
public:
DblStack(intsz=10);
//初始化双栈,总体积m的默认值为10
~DblStack(){delete[]elements;
}//析构函数
voidDblPush(constType&
x,inti);
//把x插入到栈i的栈顶
intDblPop(inti);
//退掉位于栈i栈顶的元素
Type*DblGetTop(inti);
//返回栈i栈顶元素的值
intIsEmpty(inti)const{returntop[i]==bot[i];
}//判栈i空否,空返回1,否则返回0
intIsFull()const{returntop[0]+1==top[1];
}//判栈满否,满则返回1,否则返回0
voidMakeEmpty(inti);
//清空栈i的内容
DblStack<
DblStack(intsz):
m(sz),top[0](-1),bot[0](-1),top[1](sz),bot[1](sz){
//建立一个最大尺寸为sz的空栈,若分配不成功则错误处理。
elements=newType[m];
//创建栈的数组空间
assert(elements!
=NULL);
//断言:
动态存储分配成功与否
voidDblStack<
DblPush(constType&
x,inti){
//如果IsFull(),则报错;
否则把x插入到栈i的栈顶
assert(!
IsFull());
栈满则出错处理,停止执行
if(i==0)elements[++top[0]]=x;
//栈0情形:
栈顶指针先加1,然后按此地址进栈
elseelements[--top[1]]=x;
//栈1情形:
栈顶指针先减1,然后按此地址进栈
}
intDblStack<
DblPop(inti){
//如果IsEmpty(i),则不执行退栈,返回0;
否则退掉位于栈i栈顶的元素,返回1
if(IsEmpty(i))return0;
//判栈空否,若栈空则函数返回0
if(i==0)top[0]--;
//栈0情形:
栈顶指针减1
elsetop[1]++;
//栈1情形:
栈顶指针加1
return1;
Type*DblStack<
DblGetTop(inti){
//若栈不空则函数返回该栈栈顶元素的地址。
if(IsEmpty(inti))returnNULL;
//判栈i空否,若栈空则函数返回空指针
return&
elements[top[i]];
//返回栈顶元素的值
}
template<
voidMakeEmpty(inti){
if(i==0)top[0]=bot[0]=-1;
elsetop[1]=bot[1]=m;
4-5写出下列中缀表达式的后缀形式:
(1)A*B*C
(2)-A+B-C+D
(3)A*-B+C
(4)(A+B)*D+E/(F+A*D)+C
(5)A&
&
B||!
(E>
F)/*注:
按C++的优先级*/
(6)!
(A&
!
((B<
C)||(C>
D)))||(C<
E)
(1)AB*C*
(2)A-B+C-D+
(3)AB-*C+
(4)AB+D*EFAD*+/+C+
(5)AB&
EF>
||
(6)ABC<
CD>
||!
&
CE<
4-6根据课文中给出的优先级,回答以下问题:
(1)在函数postfix中,如果表达式e含有n个运算符和分界符,问栈中最多可存入多少个元素?
(2)如果表达式e含有n个运算符,且括号嵌套的最大深度为6层,问栈中最多可存入多少个元素?
(1)在函数postfix中,如果表达式e含有n个运算符和分界符,则可能的运算对象有n+1个。
因此在利用后缀表达式求值时所用到的运算对象栈中最多可存入n+1个元素。
(2)同上。
4-7设表达式的中缀表示为a*x-b/x↑2,试利用栈将它改为后缀表示ax*bx2↑/-。
写出转换过程中栈的变化。
若设当前扫描到的运算符ch的优先级为icp(ch),该运算符进栈后的优先级为isp(ch),则可规定各个算术运算符的优先级如下表所示。
运算符
;
(
^
*,/,%
+,-
)
isp
0
1
7
5
3
8
icp
6
4
2
isp也叫做栈内(instackpriority)优先数,icp也叫做栈外(incomingpriority)优先数。
当刚扫描到的运算符ch的icp(ch)大于isp(stack)时,则ch进栈;
当刚扫描到的运算符ch的icp(ch)小于isp(stack)时,则位于栈顶的运算符退栈并输出。
从表中可知,icp(“(”)最高,但当“(”进栈后,isp(“(”)变得极低。
其它运算符进入栈中后优先数都升1,这样可体现在中缀表达式中相同优先级的运算符自左向右计算的要求。
运算符优先数相等的情况只出现在括号配对“)”或栈底的“;
”号与输入流最后的“;
”号配对时。
前者将连续退出位于栈顶的运算符,直到遇到“(”为止。
然后将“(”退栈以对消括号,后者将结束算法。
步序
扫描项
项类型
动作
栈的变化
输出
☞'
;
'
进栈,读下一符号
a
操作数
☞直接输出,读下一符号
A
*
操作符
☞isp('
'
)<
icp('
*'
),进栈,读下一符号
x
ax
-
)>
-'
),退栈输出
ax*
b
ax*b
/
/'
),进栈,读下一符号
-/
ax*bx
↑
/'
↑'
-/↑
9
ax*bx2
10
ax*bx2↑
ax*bx2↑/
ax*bx2↑/-
☞结束
4-8试利用运算符优先数法,画出对如下中缀算术表达式求值时运算符栈和运算对象栈的变化。
a+b*(c-d)-e↑f/g
设在表达式计算时各运算符的优先规则如上一题所示。
因为直接对中缀算术表达式求值时必须使用两个栈,分别对运算符和运算对象进行处理,假设命名运算符栈为OPTR(operator的缩写),运算对象栈为OPND(operand的缩写),下面给出对中缀表达式求值的一般规则:
(1)建立并初始化OPTR栈和OPND栈,然后在OPTR栈中压入一个“;
”
(2)从头扫描中缀表达式,取一字符送入ch。
(3)当ch不等于“;
”时,执行以下工作,否则结束算法。
此时在OPND栈的栈顶得到运算结果。
①如果ch是运算对象,进OPND栈,从中缀表达式取下一字符送入ch;
②如果ch是运算符,比较ch的优先级icp(ch)和OPTR栈顶运算符isp(OPTR)的优先级:
☞若icp(ch)>
isp(OPTR),则ch进OPTR栈,从中缀表达式取下一字符送入ch;
☞若icp(ch)<
isp(OPTR),则从OPND栈退出一个运算符作为第2操作数a2,再退出一个运算符作为第1操作数a1,从OPTR栈退出一个运算符θ形成运算指令(a1)θ(a2),执行结果进OPND栈;
☞若icp(ch)==isp(OPTR)且ch==“)”,则从OPTR栈退出栈顶的“(”,对消括号,然后从中缀表达式取下一字符送入ch;
根据以上规则,给出计算a+b*(c-d)-e↑f/g时两个栈的变化。
动作
OPND栈
OPTR栈
☞OPTR栈与OPND栈初始化,‘;
’进OPTR栈,
取第一个符号
a
☞a进OPND栈,取下一符号
+
☞icp(‘+’)>
isp(‘;
’),进OPTR栈,取下一符号
+
b
☞b进OPND栈,取下一符号
ab
*
☞icp(‘*’)>
isp(‘+’),进OPTR栈,取下一符号
+*
(
☞icp(‘(’)>
isp(‘*’),进OPTR栈,取下一符号
+*(
c
☞c进OPND栈,取下一符号
abc
7
-
☞icp(‘-’)>
isp(‘(’),进OPTR栈,取下一符号
+*(-
8
d
☞d进OPND栈,取下一符号
abcd
9
)
☞icp(‘)’)<
isp(‘-’),退OPND栈‘d’,退OPND
栈‘c’,退OPTR栈‘-’,计算c–d→s1,结果
进OPND栈
abs1
10
同上
☞icp(‘)’)==isp(‘(’),退OPTR栈‘(’,对消括号,
取下一符号
+*
11
☞icp(‘-’)<
isp(‘*’),退OPND栈‘s1’,退OPND
栈‘b’,退OPTR栈‘*’,计算b*s1→s2,结果
as2
+
12
isp(‘+’),退OPND栈‘s2’,退OPND
栈‘a’,退OPTR栈‘+’,计算a*s2→s3,结果
s3
13
s3
14
e
☞e进OPND栈,取下一符号
s3e
15
↑
☞icp(‘↑’)>
isp(‘-’),进OPTR栈,取下一符号
-↑
16
f
☞f进OPND栈,取下一符号
s3ef
17
/
☞icp(‘/’)<
isp(‘↑’),退OPND栈‘f’,退OPND
栈‘e’,退OPTR栈‘↑’,计算e↑f→s4,结果
s3s4
18
☞icp(‘/’)>
-/
19
g
☞g进OPND栈,取下一符号
s3s4g
20
☞icp(‘;
’)<
isp(‘/’),退OPND栈‘g’,退OPND
栈‘s4’,退OPTR栈‘/’,计算s4/g→s5,结果
s3s5
21
isp(‘-’),退OPND栈‘s5’,退OPND
栈‘s3’,退OPTR栈‘-’,计算s3–s5→s6,结
果进OPND栈
s6
22
’)==