数据结构(C语言版)习题解答

}

}

}

qa->next=A->next; A->next=qa;

//将当前最小结点插入A表表头

else{ }

qb=pb; pb=pb->next; qb->next=A->next; A->next=qb;

//将当前最小结点插入A表表头

while(pa){ }

while(pb){ } pb=B; free(pb); return OK;

qb=pb; pb=pb->next; qb->next=A->next; A->next=qb; qa=pa; pa=pa->next; qa->next=A->next; A->next=qa;

2.13 设以带头结点的双向循环链表表示的线性表L=(a1,a2,a3,...,an)。试写一时间复杂度为O(n)的算法,将L改造为L=(a1,a3,...,an,...,a4,a2)。

void Reform(DuLinkedList &L)//按1,3,5,...4,2 的顺序重排双向循环链表L 中的所有结点 { p=L.next;

while(p->next!=L&&p->next->next!=L) {

p->next=p->next->next; p=p->next;

} //p指向最后一个奇数结点

if(p->next==L) //结点个数是奇数,使最后一个奇数结点next指向最后一个偶数结点 p->next=L->pre->pre;

else//结点个数是偶数,使最后一个奇数结点next指向最后一个偶数结点 p->next=L->pre;

p=p->next; //此时p 指向最后一个偶数结点

while(p->pre->pre!=L) {

p->next=p->pre->pre; p=p->next; }

p->next=L;//最后一个结点next指向头结点

//调整了next 链的结构,此时pre 链仍为原状 //调整pre 链的结构

for(p=L;p->next!=L;p=p->next) p->next->pre=p; L->pre=p; //头结点的pre指向a2结点

}//Reform

第三章 栈和队列

3.6 试写一个算法,识别依次读入的一个以@为结束符的字符序列是否为形如“序列1&序列2”模式的字符序列。其中,序列1和序列2中都不包含字符‘&’,且序列2是序列1的逆序。例如,“a+b&b+a”是属于该模式的字符序列,而“1+3&3-1”则不是。 算法:

int SeqReverse()//判断输入的字符串中'&'前和'&'后部分是否为逆串,是则返回1,否则返回0 {

InitStack(s);

while((e=getchar())!='&') {

if(e==’@’)

return 0;//不允许在’&’之前出现’@’ push(s,e); }//序列1输入完毕 while( (e=getchar())!='@') {

if(StackEmpty(s)) return 0; pop(s,c); if(e!=c) return 0; }

if(!StackEmpty(s))

return 0;//序列1元素还有剩余 return 1;

}//IsReverse

3.7 假设一个算术表达式中可以包含三种符号:圆括号“(”和“)”、方括号“[”和“]”、花括号“{”和“}”,且这三种括号可按任意次序嵌套使用。编写判别给定表达式中所含的括号是否正确配对的算法(已知表达式已存入数据元素为字符的顺序表中)。 算法:

Status BracketTest(char *str)//判别表达式中三种括号是否匹配 {

InitStack(s); for(p=str;*p;p++) {

if(*p=='(' || *p=='[' || *p=='{' ) push(s,*p); else

if(*p==' ) ' || *p==' ] ' || *p==' } ') {

if(StackEmpty(s)) return ERROR; pop(s,c);

if(*p==')'&&c!='(') return ERROR; if(*p==']'&&c!='[') return ERROR;

if(*p=='}'&&c!='{') return ERROR; //必须与当前栈顶括号匹配 }//if }//for

if(!StackEmpty(s)) return ERROR;//进栈的符号还有剩余,Error return OK; }//BracketTest

3.8 设表达式由单字母变量、双目运算符和圆括号组成(如:“(a*(b+c)-d)/e)”。试写一个算法,将一个书写正确的表达式转换为逆波兰式。

思路:

1.遇到数字直接发送 2.遇到’(’直接入栈 3.遇到’)’则将栈内元素发送直至遇到’(’ 4.栈空则直接入栈 5.栈非空时若优先级大于栈内则入栈,否则栈内元素出栈

int RankOfOperator(char c){ switch(c){

case '#': return -1; case '(': return 0; case '+':

case '-': return 1;

case '*': case '/':

case ')':return 2; } }

int Precede(char c, char ch){

return RankOfOperator(c)>RankOfOperator(ch); }

void ExpressionTOPolandStyle(char suffix[],char *exp){ Stack s; InitStack(s,100); int i=0; char c;

while(*exp){

if(isdigital(*exp)) suffix[i++]=*exp; else{

switch(*exp){

case '(': push(s,'(');

case ')': while((c=pops(s))!='(') suffix[i++]=c; break;

default: if(IsEmpty(s)) push(s,*exp); else{

suffix[i++]=pop(s); exp--;

//与后面的exp++相抵消,使得栈内优先级大于等于栈外的都出栈

}

}

}//end switch }//end else exp++; }//end while

while(!IsEmpty(s)) suffix[i++]=pop(s); suffix[i]=0;

3.10 假设以带头结点的单循环链表表示队列,只设一个尾指针指向队尾元素,不设头指针。试编写相应的队列初始化、入队和出队算法(在出队算法中要传回队头元素的值)

要点:定义好数据类型,带头结点的单循环链表,只有尾指针,注意删除元素时只有一个元素的特殊性

typedef int DataType

联系客服:779662525#qq.com(#替换为@) 苏ICP备20003344号-4