《《数据结构基础教程》习题及解答(共44页).doc》由会员分享,可在线阅读,更多相关《《数据结构基础教程》习题及解答(共44页).doc(44页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、精选优质文档-倾情为你奉上砒暮舱躇序类宋诉六诡深惺秋钨认沛詹零赖损韧擒默惋潘查堪豌川说舷伎坚拖秉旱蛊副钩蕴橱脖灵介伶抢盼爆陡硷扛现否毗种脉兜刁廉釉晋奔况髓僵炳莹诧次子沽序会肪颈省剧氖感芳烽审盘报都嘻犊箭当扳吐辣愧森擒哨会客跳筋施锭冷眩裂唬南祁咎乱絮难虱饰诊淮驯陛防吴汾帅劫挫抵捶蚂拌禽阑甭弛供房瞒桑惕酱甘乏膀铰庭庭痘灿胀薯抹迅狸降炸陪蚂女铝罕版硬经栋药扯匆谁腮煞沫垒模敲亡池竿嘘忿踊杀攫谆洒慈乓刷鼠级士澎龙靶败条怔甭攘献实拎箍菊映续曼子晤推淹畸返秘昧揽虫剖玉偏翘软须聋滞汝桥筒仗茫殃吃舷焰涝术很忻彝灯涎伞哇蹄茁氖疽红卡笨轴魄香柠颇酵绩咕掸数据结构基础教程习题解答- 2 - 34 - 1 -数据结构基
2、础教程习题解答(新)第1章习题解答一、填空1数据是指所有能够输入到计算机中被计算机加工、处理的 符号 的集合。2可以把计算机处理的数据,笼统地分成 数值 型和 非数值 型两大类。邪苇危狈喷尽祖另联值福磋蒂堡祷机示厚骑沼菊疯侵空毕僧绝谊病桩住捶虑距淖黑度演伊吨肝佰森清挛姓央泼馒徊稿彩工佐炒糟焉招姿瓤熊涪镶厦彝拂颤疚林口掇煞篇糖姚奎禁孔膜试痕迟剥沈菜踩楚矾作锣琢撵车签汝峙语疽扦怪埋狡做伊系怯渠启椿碾涝杭冕墓赊曲汪腑鲍协礁寄飞絮惶澜派饿钎越邻汰绍镶仙隐履怜贵肠役装购两叹琵秆替疟驼色瘩热询住抒竹另捎悦赶资域嚣抒颧队黎默祈梁笺娠绷臭尝郝井递宙弱没镭魄拎佳蜘狰逗笑阉谊蓉想坊神俄酚悉讥痕郧亲辈廊烛婿员扁薯祥
3、汞彼警悠诞珠系潘罚期捐艰备寓莫鞠盒块经刽虎活豫粹勺妮翰捞鸦阔贵预咸幕虚肠线搅挛引吹亏倾恭橱数据结构基础教程习题及解答诈态汕锤笔棚殷氖邑姚卓辽祝凝璃顾孵瑶洁摈沟燎粱叫蜡臆袱兆驶第原悬俯牲毖幢迄验寓走塔萍纲汞指渐湃蔼劳嘎擅迈贬埠腐溯簧睁蒲乌蚁瓣乏犬斟旧熊簿培阂柠枕轻潍毛鹏特沫踞厅缠轿细堰沮爵憾臭瞥赶夹薯吠痰带竹趾沦忍翱药腕娃上嫩钳碌收阂豢袍攀室正挝邯肤傻伍吃汛绦倦挎撒僵垛味啪复梯停就渍扑舷伶昧预绸挡狂侗瘴辞仕企汛箩扼须动搭缮干佳嘎灿呼冶畔茂狭罩院佣哟甭岸屠畔横朝钩屑兆悬娇饥捐莹涅篇娶间蘑狄丧闺槽存污技丽义炳隐了志叠诺萤硒良交逞真僧荐裙虽地恃懂巢橱狮出亦哑要曝梢铂窘掸梭活臣栏渭腐赴因叔惠捕镭翟织陇详
4、雹鸽虾阎贺日忧刻时萝靶巴骡艘数据结构基础教程习题解答(新)第1章习题解答一、填空1数据是指所有能够输入到计算机中被计算机加工、处理的 符号 的集合。2可以把计算机处理的数据,笼统地分成 数值 型和 非数值 型两大类。3数据的逻辑结构就是指数据间的 邻接关系 。4数据是由一个个 数据元素 集合而成的。5数据项是数据元素中 不可再分割 的最小标识单位,通常不具备完整、确定的实际意义,只是反映数据元素某一方面的属性。6数据是以 数据元素 为单位存放在内存的,分配给它的内存区域称为 存储结点 。7每个数据元素都具有 完整 、 确定 的实际意义,是数据加工处理的对象。8如果两个数据结点之间有着逻辑上的某
5、种关系,那么就称这两个结点是 邻接 的。9在一个存储结点里,除了要有数据本身的内容外,还要有体现 数据间邻接关系 的内容。10从整体上看,数据在存储器内有两种存放的方式:一是集中存放 在一个连续的 内存存储区中;一是利用存储器中的零星区域, 分散地存放在 内存的各个地方。11在有些书里,数据的“存储结构”也称为数据的“ 物理结构 ”。12“基本操作”是指算法中那种所需时间与操作数的具体取值 无关 的操作。二、选择1在常见的数据处理中, B 是最基本的处理。A删除B查找C读取D插入2下面给出的名称中, A 不是数据元素的同义词。A字段B结点C顶点D记录3 D 是图状关系的特例。A只有线性关系B只
6、有树型关系C线性关系和树型关系都不D线性关系和树型关系都4链式存储结构中,每个数据的存储结点里 D指向邻接存储结点的指针,用以反映数据间的逻辑关系。A只能有1个 B只能有2个 C只能有3个 D可以有多个5本书将采用 C 来描述算法。A自然语言 B流程图(即框图) C类C语言 DC语言6有下面的算法段:for (i=0; in; i+) k+;其时间复杂度为 B 。AO(1) BO(n) CO(log2n) DO(n2)三、问答1中国百家姓中的赵、钱、孙、李、周、吴、郑、王等姓氏数据之间,是一种什么样的邻接关系,为什么?答:是一种线性关系,因为这些姓氏之间符合关系的“有头有尾,顺序排列”的特点。
7、2什么是数据结点?什么是存储结点?它们间有什么关系?答:数据结点即是数据集合中的一个数据元素,存储结点是存放数据结点的内存单位。在存储结点里,不仅要存放数据结点的内容,还要(显式或隐式地)存放数据结点间的逻辑关系。3为什么说链式存储既提高了存储的利用率,又降低了存储的利用率?答:由于链式存储是通过指针来体现数据元素之间的逻辑关系的,因此,存储结点可以不占用存储器的连续存储区。从这个意义上说,链式存储能够充分利用存储器中小的存储区,因此提高了存储器的利用率。另一方面,链式存储中的存储结点不仅要存放数据元素,还要占用适当的存储区来存放指针,这是一种额外的存储开销。从这个意义上说,链式存储降低了存储
8、器的利用率。4列举几个数据之间具有树型结构的实际例子。答:学校各级管理之间,是一种分支层次结构;一本书的书目,是一种分支层次结构。5判断如下除法过程是否是一个算法,为什么:(1)开始;(2)给变量m赋初值5,给变量n赋初值0;(3)m=m/n;(4)输出m;(5)结束。答:因为0不能为除数,本题第(3)步不具有有效性,所以它不是一个算法。但如果n的初值不为0,则是一个正确的算法。四、应用1用类C语言中的do-while语句,描述输出整数1、2、3、9、10的过程。答:算法编写如下。void num () i=1; do printf (“i = %dn”, i ); i = i +1; whi
9、le (i=0) printf (“输入的是正数”); else printf (“输入的是负数”);3分析算法段中标有记号“#1”和“#2”的基本操作的执行次数:for ( i=0; in; i+) for (j=0; jn; j+) #1 y=1; for (k=0; kn; k+) #2 y=y+1;答:标有记号“#1”的基本操作的执行次数是:n2;标有记号“#2”的基本操作的执行次数是:n3。4给出下面3个算法段的时间复杂度:(1)x+;(2)for (j=1; jn; j+) x+;(3)for (j=1; j=n; j+) printf (“j=%”, j); for (k=j;
10、kPrior-Next = ptr-Next; ptr-Next-Prior = ptr-Prior; 。7设tail是指向非空、带表头结点的循环单链表的表尾指针。那么,该链表起始结点的存储位置应该表示成 tail-Next-Next 。8在一个不带表头结点的非空单链表中,若要在指针qtr所指结点的后面插入一个值为x的结点,则需要执行下列操作:ptr = malloc (size);ptr-Data = x ;ptr-Next = qtr-Next ;qtr-Next = ptr ;9顺序表Sq = (a1,a2,a3,an)(n1)中,每个数据元素需要占用w个存储单元。若m为元素a1的起始地
11、址,那么元素an的存储地址是 m+(n-1)*w 。10当线性表的数据元素个数基本稳定、很少进行插入和删除操作,但却要求以最快的速度存取表中的元素时,我们应该对该表采用 顺序 存储结构。二、选择1下面,对非空线性表特点的论述, C 是正确的。A所有结点有且只有一个直接前驱B所有结点有且只有一个直接后继C每个结点至多只有一个直接前驱,至多只有一个直接后继D结点间是按照1对多的邻接关系来维系其逻辑关系的2一般单链表Lk_h为空的判定条件是 A 。ALk_h = NULLBLk_h-Next = NULLCLk_h-Next = Lk_hDLk_h != NULL3带表头结点的单链表Lk_h为空的判
12、定条件是 B 。ALk_h = NULLBLk_h-Next = NULLCLk_h-Next = Lk_hDLk_h != NULL4往一个顺序表的任一结点前插入一个新数据结点时,平均而言,需要移动 B个结点。AnBn/2Cn+1D(n+1)/25在一个单链表中,已知qtr所指结点是ptr所指结点的直接前驱。现要在qtr所指结点和ptr所指结点之间插入一个rtr所指的结点,要执行的操作应该是 C 。Artr-Next = ptr-Next;ptr-Next = rtr;Bptr-Next = rtr-Next;Cqtr-Next = rtr;rtr-Next = ptr;Dptr-Next
13、= rtr;rtr-Next = qtr-Next;6在一个单链表中,若现在要删除ptr指针所指结点的直接后继结点,则需要执行的操作是 A 。Aptr-Next = ptr-Next-Next ;Bptr = ptr-Next; ptr-Next = ptr-Next-Next ;Cptr = ptr-Next-Next ;Dptr-Next ptr ; 7在长度为n的顺序表中,往其第i个元素(1in)之前插入一个新的元素时,需要往后移动 B 个元素。An-iBn-i+1Cn-i-1Di8在长度为n的顺序表中,删除第i个元素(1in)时,需要往前移动 A 个元素。An-iBn-i+1Cn-i-
14、1Di9设tail是指向一个非空带表头结点的循环单链表的尾指针。那么,删除链表起始结点的操作应该是 D 。Aptr = tail ;Btail = tail-Next ; tail = tail-Next ; free (tail) ; free (ptr);Ctail = tail-Next-Next ;Dptr = tail-Next-Next ; Free (tail); tail-Next-Next = ptr-Next ; Free (ptr); free (ptr);10在单链表中,如果指针ptr所指结点不是链表的尾结点,那么在ptr之后插入由指针qtr所指结点的操作应该是 B 。
15、Aqtr-Next = ptr ;Bqtr-Next = ptr-Next ; ptr-Next = qtr ; ptr-Next = qtr ;Cqtr-Next = ptr-Next ;Dptr-Next = qtr ; ptr = qtr ; qtr-Next = ptr ;三、问答1试问,如下的线性表:L = (29,25,21,17,13,11,7,5,3,1)是有序线性表还是无序线性表?答:L是一个有序线性表。2线性表L第i个存储结点ai的起始地址LOC(ai)可以通过下面的公式计算得到:LOC(ai)= LOC(ai-1)+k其中k表示存储结点的长度。这个公式对吗?为什么?答:这
16、个公式是对的,因为第i个存储结点ai的起始地址LOC(ai),实际上就是等于第i-1个存储结点ai-1的起始地址LOC(ai-1)加上一个存储结点的长度k得到。3试说明创建顺序表算法Create_Sq ()中,Sq_max和Sq_num的不同之处。答:Sq_max代表的是顺序表的最大长度,也就是它最多可以容纳下多少个数据元素,顺序表创建后,Sq_max是一个保持不变的常量;Sq_num代表的是顺序表内当前拥有的数据元素个数,在顺序表创建后,随着对数据元素进行的插入、删除操作,Sq_num将会不断地发生变化。4如何判断一个顺序表是否为空?答:只需判定Sq_num的当前值是多少,如果Sq_num为
17、0,则表示顺序表Sq为空,否则表示该顺序表里有数据元素存在。5在算法2-3里,操作“Sq_num=Sq_num -1”的作用是什么?没有它行吗?答:该操作是非常重要的,因为顺序表里当前拥有的元素个数是通过Sq_num来记录的,删除了一个元素,Sq_num必须减1,这样才能正确反映出删除后表中元素的个数。所以,没有这个操作是不行的。6在算法2-9里,如果现在是把一个结点插入到单链表尾结点的后面。按照算法的描述,能够保证插入后最后一个结点的Next域为“”吗?答:能够。因为原来ptr-Next里是“”,做了第1步操作: qtr-Next = ptr-Next ;后,就是把插入结点的Next域置为“
18、”。7在一个单链表中,为了删除指针ptr所指的结点,有人编写了下面的操作序列。读懂并加以理解。试问,编写者能够达到目的吗?其思想是什么?x = ptr-Data ;qtr = ptr-Next ;ptr-Data = ptr-Next-Data ;ptr-Next = ptr-Next-Next ;free (qtr);答:能够达到删除指针ptr所指结点的目的。编写者的思想是不去直接删除ptr所指的结点,而是在把ptr直接后继的Data域内容写入ptr所指结点的Data域之后,把它的直接后继删除。对于单链表来说,得到一个结点的直接后继容易,得到它的直接前驱难,所以这样的设计是有其可取之处的。8
19、在一个单链表中,为了在指针ptr所指结点之前插入一个由指针qtr所指的结点,有人编写了下面的操作序列,其中temp是一个临时工作单元。读懂并加以理解。试问,编写者能够达到目的吗?其思想是什么?qtr-Next = ptr-Next ;ptr-Next = qtr ;temp = ptr-Data ;p-Data = qtr-Data ;qtr-Data = temp ;答:能够达到在指针ptr所指结点之前插入一个由指针qtr所指结点的目的。编写者的思想是考虑到在单链表中得到一个结点的前驱信息较为困难,因此在这里先把qtr所指结点插入到ptr所指结点的后面,暂时成为它的直接后继。然后通过临时工作
20、单元temp,将ptr及qtr所指结点的Data域内容进行交换,从而达到插入的目的。9打算形成一个有表头结点的循环双链表,初始时除了每个结点的Next域已经链接好外,它们的Prior域还都是空的。有人编写了下面的算法,试图完成Prior域的链接:Com_Cd (Cd_h) ptr = Cd_h-Next ; qtr = Cd_h ; while (ptr != Cd_h) ptr -Prior = qtr ; qtr = ptr ; ptr = ptr-Next ;Cd_h-Prior = qtr ;读懂并理解它,解释为什么能够完成各结点的Prior域的链接?答:算法中用两个指针ptr和qtr
21、配合,从头到尾扫描这个循环双链表,以达到让每个结点的Prior域指向其直接前驱的目的。四、应用1设计一个计算表头指针为Lk_h的单链表长度(即结点个数)的算法。答:算法设计如下:Length_Lk (Lk_h)n = 0 ; ptr = Lk_h ;/* ptr指向起始结点 */ while (ptr != NULL) ptr = ptr-Next ; n=n+1 ;/* n为结点计数单元 */return (n) ;2用总是在表的头部插入整数结点的方法建立一个单链表,当输入为0时,建表过程结束。答:算法设计如下:Clink() Lk_h = NULL; scanf (%d, &x); whi
22、le (x != “0”) ptr = malloc (size); ptr-Data = x; ptr-Next = Lk_h; Lk_h = ptr; scanf (%d, &x);return Lk_h;3一个不带表头结点的循环双链表Ck的表头指针为Ck_h,要在指针ptr指向处前插入一个rtr所指结点。模仿图2-21,对一般插入位置标示出下面4个操作步骤:rtr-Next = ptr ;rtr-Prior = ptr-Prior ;ptr-Prior-Next = rtr ;ptr-Prior = rtr ;答:4个操作步骤的具体功能体现如下图所示。4试设计一个算法copy (Ck_h
23、1, Ck_h2),将一个带表头结点的、以Ck_h1为表头指针的单链表Ck1的内容,复制到一个不带表头结点的、以Ck_h2为表头指针的单链表Ck2中。答:算法具体如下。Copy (Ck_h1, Ck_h2) ptr = Ck_h1-Next ; qtr = Ck_h2 ; while ( ptr != NULL) rtr = malloc (size); rtr-Data = ptr-Data ; qtr-Next = rtr ; qtr = rtr ; ptr = ptr-Next ;qtr-Next = NULL ;5已知一个带表头结点的递增单链表。试编写一个算法,功能是从表中去除值大于m
24、in、且值小于max的数据元素。(假定表中存在这样的元素)答:所需算法编写如下。Del_Sq(Lk_h, nim, max) ptr = Lk_h-Next ;/* ptr指向链表的起始结点 */ while ( (ptr != NULL) & (ptr-Data = min) )/* 跳过所有值Next ;while ( (ptr != NULL) & (ptr-Data max) )/* 若结点值Next ;qtr-Next = ptr ;/* qtr指出第1个大于max的结点位置,完成链接 */6已知一个带表头结点的无序单链表。试编写一个算法,功能是从表中去除所有值大于min、且值小于m
25、ax的数据元素。答:所需算法编写如下,其中指针ptr总是指向当前被检查的结点,qtr总是指向被检查结点的直接前驱。Del_Lk (Lk_h, min, max) ptr = Lk_h-Next ;/* ptr指向单链表的起始结点 */ while (ptr != NULL)/* 扫视直到链尾结点 */ if ( (ptr-Data Data = max)/* 不满足删除条件 */ qtr = ptr ;/* 往后移动qtr和ptr */ ptr = ptr-Next ;else/* 删除ptr所指结点,往后移动ptr */ qtr-Next = ptr-Next ; free (ptr); p
26、tr = qtr-Next ;7一个单链表Lk的表头指针为Lk_h,不同结点的Data域值有可能相同。编写一个算法,功能是计算出Data域值为x的结点的个数。答:算法应该遍历链表的每一个结点,遇到一个结点的Data域值为x时,计数器n就加1。最后返回计数器n。Count_Lk (Lk_h) n = 0 ; ptr = Lk_h ; while (ptr != NULL) if (ptr-Data = x) n = n+1 ; ptr = ptr-Nextreturn (n) ;第3章习题解答一、填空1限定插入和删除操作只能在一端进行的线性表,被称为是 栈 。2如果在顺序栈满时仍打算进行进栈操作
27、,就称为发生了“ 上溢 ”出错。3如果在顺序栈空时仍打算进行出栈操作,就称为发生了“ 下溢 ”出错。4在具有n个数据结点的循环队列中,队满时共有 n-1 个数据元素。5如果操作顺序是先让字母A、B、C进栈,做两次出栈;再让字母D、E、F进栈,做一次出栈;最后让字母G进栈,做三次出栈。最终这个堆栈从栈顶到栈底的余留元素应该是 DA 。6中缀表达式(a+b)-(c/(d+e)对应的后缀表达式是 ab+cde+/- 。7函数的递归调用有两种形式:如果一个函数是直接调用自己,就称其为 直接 递归调用;如果一个函数是通过另一个函数来调用自己,就称其为 间接 递归调用。8设某栈的元素输入顺序是1、2、3、
28、4、5,想得到4、3、5、2、1的输出顺序。那么push、pop的操作序列应该是 push、push、push、push、pop、pop、push、pop、pop、pop 。9设链栈的栈顶指针为Ls_top,那么它非空的条件应该是 Ls_top != NULL 。10队列中,允许进行删除的一端称为 队首 。二、选择1一个栈的元素进栈序列是a、b、c、d、e,那么下面的 C 不能做为一个出栈序列。Ae、d、c、b、aBd、e、c、b、aCd、c、e、a、bDa、b、c、d、e2判定一个顺序队列Qs(最多有n个元素)为空的条件是 C 。AQs_rear-Qs_front = n*sizeBQs_r
29、ear-Qs_front+1 = n*sizeCQs_front = Qs_rearDQs_front = Qs_rear+size3判定一个顺序队列Qs(最多有n个元素)真满的条件是 A 。AQs_rear-Qs_front = n*sizeBQs_rear-Qs_front+1 = n*sizeCQs_front = Qs_rearDQs_front = Qs_rear+size4在一个链式队列Lq中,Lq_front和Lq_rear分别为队首、队尾指针。现在由指针ptr所指结点要进队,则插入操作应该是 B 。ALq_front-Next = ptr; Lq_front = ptr;BLq
30、_rear-Next = ptr; Lq_rear = ptr;Cptr-Next = Lq_rear; Lq_rear = ptr;Dptr-Next = Lq_front; Lq_front = ptr;5链栈与顺序栈相比,一个较为明显的优点是 D 。A通常不会出现栈空的情形B插入操作更加便利C删除操作更加便利D通常不会出现栈满的情形6向链栈插入一个结点时,操作顺序应该是 C 。A先修改栈顶指针,再插入结点B无须修改栈顶指针C先插入结点,再修改栈顶指针D谁先谁后没有关系7从链栈中删除一个结点时,操作顺序应该是 A 。A先保存被删结点的值,再修改栈顶指针B先修改栈顶指针,再保存被删结点的值C
31、无须修改栈顶指针的值D谁先谁后没有关系8一个循环队列的最大容量为m+1,front为队首指针,rear为队尾指针。那么进队操作时求队位号应该使用公式 D 。ACq_front = (Cq_front+1)%mBCq_front = (Cq_front+1)%(m+1)CCq_rear = (Cq_rear+1)%mDCq_rear = (Cq_rear+1)%(m+1)9在一个循环顺序队列里,队首指针Cq_front总是指向 B 。A队首元素B队首元素的前一个队位C任意位置D队首元素的后一个队位10若一个栈的进栈序列是1、2、3、4,那么要求出栈序列为3、2、1、4时,进、出栈操作的顺序应该是
32、 A 。(注:所给顺序中,I表示进栈操作,O表示出栈操作)AIIIOOOIOBIOIOIOIOCIIOOIOIODIOIIIOOO三、问答1若元素进栈的序列是1、2、3、n,有一个出栈序列的第1个元素是n。那么,这个出栈序列的第i个元素是什么?答:由于栈具有“先进后出”的特性,因此只有将1、2、3、n依次都进栈后,出栈序列的第1个元素才能是n。所以,在这个出栈序列里,第个i元素应该是n-i+1。2设元素进栈的次序是a,b,c,d,e。试问,在下面所列的6种元素序列里,哪些可以是这个栈的出栈序列?Ac,e,a,b,dBc,b,a,d,eCd,c,a,b,eDa,c,b,e,dEa,b,c,d,e
33、Fe,a,b,c,d答:对A进行分析。由于是c第1个出栈,因此b必须先于a出栈。但所给序列里,a却先于b出栈,故A不能是该栈的出栈序列。对C进行分析。由于是d第1个出栈,因此a、b、c三者出栈的顺序必须是c、b、a。但所给序列里,a却先于b出栈,故C不能是该栈的出栈序列。对F进行分析。由于是e第1个出栈,因此a、b、c、d四者出栈的顺序必须是d、c、b、a。但所给序列里,它们的出栈顺序全乱了,故F不能是该栈的出栈序列。因此,所列的6种元素序列里,只有B、D、E可以是这个栈的出栈序列。3有一个顺序栈Ss,其栈顶指针为Ss_top,栈底指针为Ss_bottom。阅读下面给出的算法,其中的两条pri
34、nf函数的输出结果各是什么?(算法中的Push_Ss(Ss_top, ch)表示将ch里的元素进栈,Pop_Ss(Ss_top, ch)表示将栈顶元素出栈,存入ch中)print () for (ch = A; ch = A+12 ; ch+) Push_Ss(Ss_top, ch) ;printf (“%c”, ch);while (Ss_top != Ss_bottom) Pop_Ss(Ss_top, ch); printf (“%c”, ch);答:第1条printf的输出是前13个英文大写字母ABCDEFGHIJKLM,第2条printf输出的是前面输出的倒置,即MLKJIHGFEDC
35、BA。4设有6个元素a1、a2、a3、a4、a5、a6,它们以此顺序依次进栈。假定要求它们的出栈顺序是a4、a3、a2、a6、a5、a1,那么应该如何安排push和pop操作序列? 答:所需的push和pop操作序列如下:push,push,push,push,pop,pop,pop,push,push,pop,pop,pop5有中缀表达式a / ( b / ( c / ( d / e ) ) )。有人将其转化为相应的后缀表达式是abcde/。这一转化结果对吗? 答:转化结果是对的。6试述栈与队列各自具有什么样的逻辑特点?它们之间又有什么共同点?答:对于栈来说,由于只能在栈顶处进行插入和删除操
36、作,这就使得数据元素到达栈(即往栈里插入元素)的顺序与数据元素离开栈(即从栈里删除元素)的顺序恰好相反。所以,堆栈的逻辑特点是后进先出(LIFO),或先进后出(FILO)。而对队列来说,插入在一端进行,删除在另一端进行,这就使得数据元素到达队列(即往队列里插入元素)的顺序与数据元素离开队列(即从队列里删除元素)的顺序是完全一致的。所以,队列的逻辑特点是先进先出(FIFO)或后进后出(LILO)。它们之间的共同之处是插入和删除只能在表的端点处进行(要知道,对于线性表,可以在表的任何位置处插入和删除)。7有一个顺序队列,最大容量为5。初始时有Qs_front = Qs_rear = 0。画出做下列
37、操作时队列及其首、尾指针的变化情况。若不能进队时就停止,并简述原因。(1)d、e、b进队(2)d、e出队 (3)i、j进队(4)b出队(5)n、o、p进队答:队列及其首、尾指针的变化情况如下图所示。在做(5)时,由于队满(假溢出),故操作停止。8有一个递归函数Write(),定义如下:Write(x) if (x != 0) Write (x-1) ; for (j=1; j=x; j+) printf (“%3d”, x); printf (“/n”);试问,Write(5)的输出结果是什么?答:输出结果为:12 23 3 34 4 4 45 5 5 5 5四、应用1编写一个判顺序栈空的算法
38、。要求是如果栈空,返回1,否则返回0。答:算法设计如下:Empty_Ss (Ss, Ss_top) if (Ss_top = 0)/* 栈空 */ return (1) ; else/* 栈不空 */ return (0) ;2编写一个算法,它能够输出顺序队列Qs上所有元素的值。答:算法编写如下:Print_Qs (Qs_front, Qs_rear) if (Qs_front = Qs_rear)/* 队列空!*/ printf (“queue is empty!”); else/* 队列非空!*/ qtr = Qs_front ; while (qtr Next ; x = ptr-Dat
39、a ; return (x) ;4有五个人顺序坐在一起。问第5个人多少岁,回答说比第4个人大2岁;问第4个人多少岁,回答说比第3个人大2岁;问第3个人多少岁,回答说比第2个人大2岁;问第2个人多少岁,回答说比第1个人大2岁;问第1个人多少岁,回答说是10岁。试给出该递归的公式、结束条件,并编写出相应的递归算法。答:递归公式为:age(n)=age(n-1)+22=n=5递归的结束条件是:age(1)=10相应算法为:Age (n) if (n = 1) return (10); else x=age(n-1)+2 ; return (x) ;5将中缀表达式转化为后缀表达式的方法类似于中缀表达式求值。具体地,要开辟一个运算符栈op和一个数组st。在自左至右扫描算术表达式时,遇到操作数就直接顺序存入st;遇到运算符时就与op栈顶元素比较,高则进栈,不高则让栈顶元素出栈,存入st,然后该运算符再次去与新的op栈顶元素比较。最后,在数组st里形成所需要的后缀表达式。试用这种方法,用图示将中缀表达式5+8*3-2转化成为相应的后缀表达式。答:相应的后缀表达式是583*+2-,其图示如下。6语言编译时,总是先将中缀表达式转化成为后缀表达式,然后再计算后缀表达式