数据结构(C语言)第二版 第二章课后答案

数据结构(C语言)第二版 第二章课后答案

1~5 B A B A D
6~10 B C A B D
11~15 C D D A C

1. 选择题

(1) 顺序表中第一个元素的存储地址是100, 每个元素的长度为2,则第5个元素的地址是(B )。
A.110 B.108
C.100 D.120

顺序表中的数据连续存储
要求的地址=第一个元素的地址+(i-1)个元素 × 每个元素的长度

第5 个元素的地址为:100+2*4=108 。

(2)在含n个结点的顺序表中,算法的时间复杂度是0(1)的操作是(A)。
A. 访问第 i 个结点(1 ≤ i ≤ n) 和求第i个结点的直接前驱(2 ≤ i ≤ n)
B. 在第 i 个结点后插入一个新结点(1 ≤ i ≤ n)
C. 删除第 1 个结点(1 ≤ i ≤ n)
D. 将n个结点从小到大排序

在顺序表中插入一个结点的时间复杂度都是O(n2)
排序的时间复杂度为O(n2 )或O(n log2 n)。
顺序表是一种随机存取结构,访问第 i 个结点和求第 i 个结点的直接前驱都可以直接通过数组的下标直接定位,时间复杂度是O(1) 。

(3)在一个有127个元素的顺序表中插入一个新元素并保持原来顺序不变,平均要移动的元素个数为(B )。
A.8 B.63.5
C. 63 D.7

平均要移动元素的次数

平均要移动的元素个数为: n/2
即为127/2=63.5

(4) 链接存储的存储结构所占存储空间( A)。
A. 分为两部分,一部分存放结点值,另一部分存放表示结点间关系的指针
B. 只有一部分, 存放结点值
C. 只有一部分,存储表示结点间关系的指针
D. 分两部分, 一部分存放结点值,另一部分存放结点所占单元数

链接存储的存储结构:

数据域:存储元素数值数据
指针域:存储直接后继结点的存储位置

(5)线性表若采用链式存储结构,要求内存中可用存储单元的地址(D )。
A. 必须是连续的 B. 部分地址必须是连续的
C. 一定是不连续的 D. 连续或不连续都可以

链式存储结构特点:用一组任意的存储单元存储线性表的数据元素(这组存储单元可以连续,也可以不连续)。

(6) 线性表L 在(B)情况下适用于使用链式结构实现。
A. 需经常修改L中的结点值 B. 需不断对L进行删除、插入
C. L中含有大量的结点 D. L中结点结构复杂

链表最大的优点在于插入和删除时不需要移动数据,直接修改指针即可。

(7) 单链表的存储密度(C)。
A.大于 1 B.等于 1
C.小于 1 D.不能确定

顺序表:不用为表示结点间的逻辑关系而增加额外的存储开销,存储密度等于 1
链表:需要借助指针来体现元素间的逻辑关系,存储密度小于 1

(8) 将两个各有n个元素的有序表归并成一个有序表, 其最少的比较次数是(A)。
A. n
B. 2n - 1
C. 2n
D. n-1

当第一个有序表中所有的元素都小于(或大于)第二个表中的元素,只需要用第二个表中的第一个元素依次与第一个表的元素比较,总计比较n次。

(9)在一个长度为n的顺序表中,在第i个元素(1 ≤ i ≤ n+1) 之前插入一个新元素时需向后移动(B) 个元素。
A. n-i
B. n - i + 1
C. n -i-1
D. i

第i个元素(1 ≤ i ≤ n+1) 之前插入一个新元素
需要将i和i之后所有的元素向后移动
共移动 n-i+1 个元素

(10) 线性表L=(a1 , a2, …, an), 下列陈述正确的是(D)。
A. 每个元素都有一个直接前驱和一个直接后继
B. 线性表中至少有一个元素
C. 表中诸元素的排列必须是由小到大或由大到小
D. 除第一个和最后一个元素外, 其余每个元素都有一个且仅有一个直接前驱和直接后继

表头元素没有直接前驱,表尾元素没有直接后继
线性表可以为空表
没有元素排序的限制

(11) 创建一个包括n个结点的有序单链表的时间复杂度是(C)。
A. 0(1) B. O(n)
C. O( n2 ) D. O(n log2 n)

单链表创建的时间复杂度是O(n)
而要建立一个有序的单链表,则每生成一个新结点时需要和已有的结点进行比较,确定合适的插入位置,所以时间复杂度是O(n2) 。

(12) 以下陈述错误的是(D)。
A. 求表长、定位这两种运算在采用顺序存储结构时实现的效率不比采用链式存储
结构时实现的效率低
B. 顺序存储的线性表可以随机存取
C. 由千顺序存储要求连续的存储区域, 所以在存储管理上不够灵活
D. 线性表的链式存储结构优于顺序存储结构

链式存储结构和顺序存储结构各有优缺点,有各自的不同适用场合。

(13) 在单链表中, 要将s所指结点插入到p所指结点之后, 其语句应为(D)。
A. s->next = p + 1; p->next = s;
B . (*p).next= s ; (*s).next= (*p).next;
C. s->next = p->next; p->next = s->next;
D. s->next = p->next; p->next = s;

s所指结点插入到p所指结点之后
大致就是p的下一个结点是s,s还要连接p之后原有的结点
所以先将s的下一个结点指向p的下一个结点
p的下一个结点再指向s
即:s->next = p->next; p->next = s;

(14) 在双向链表存储结构中, 删除p所指结点时修改指针的操作为(A)。
A . p->next->prior= p ->proir;p ->prior->next= p ->next;
B. p->next = p->next->next; p->next->prior = p;
C. p->prior->next= p ;p ->prior=p ->prior->prior;
D. p->prior= p ->next->next;p ->next= p ->prior->prior;

双向链表的每个数据结点中都有两个指针,分别指向直接后继和直接前驱。
删除p
就是将p的下一个结点的前驱改为p的前驱
p的前一个结点的后继改为p的后继
p->next->prior= p ->proir;p ->prior->next= p ->next;

(15)在双向循环链表中,在p指针所指的结点后插入q 所指向的新结点,其修改指针的操作是(C)。
A. p->next = q; q->prior = p; p->next->prior = q; q->next = q;
B. p->next = q; p->next->prior = q; q->prior=p; q->next = p->next;
C. q->prior = p; q->next = p->next; p->next->prior = q; p->next = q;
D. q->prior = p; q->next = p->next; p->next = q; p->next->prior = q;

在p指针所指的结点后插入q
在这里插入图片描述
q的前驱改为p q的后继改为p的后继
p的下一个结点的前驱改为p p的下一个结点改为q
q->prior = p; q->next = p->next; p->next->prior = q; p->next = q;

2. 算法设计题

(1)将两个递增的有序链表合并为一个递增的有序链表。要求结果链表仍使用原来两个链表的存储空间,不另外占用其他的存储空间。表中不允许有重复的数据。

[题目分析]
合并后的新表使用头指针Lc指向,pa和pb分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点,从第一个结点开始进行比较,当两个链表La 和Lb 均为到达表尾结点时,依次摘取其中较小者重新链接在Lc 表的最后。如果两个表中的元素相等,只摘取La表中的元素,删除Lb 表中的元素,这样确保合并后表中无重复的元素。当一个表到达表尾结点,为空时,将非空表的剩余元素直接链接在Lc 表的最后。

[ 算法描述]

void MergeList(LinkList &La,LinkList &Lb,LinkList &Lc)
{// 合并链表La 和Lb,合并后的新表使用头指针Lc 指向
	pa=La->next; pb=Lb->next;
	//pa和pb分别是链表La和Lb的工作指针, 初始化为相应链表的第一个结点
	Lc=pc=La; 				// 用La的头结点作为Lc的头结点
	while(pa && pb){		// 取较小者La中的元素,将pa链接在pc的后面,pa指针后移
		if(pa->data<pb->data){
			pc->next=pa;
			pc=pa;
			pa=pa->next;
		}else if(pa->data>pb->data) {	// 取较小者Lb中的元素,将pb链接在pc的后面,pb指针后移
			pc->next=pb; 
			pc=pb; 
			pb=pb->next;
		}else{			 // 相等时取La中的元素,删除Lb中的元素
			pc->next=pa;
			pc=pa;
			pa=pa->next;
			q=pb->next;
			delete pb;
			pb =q;
		}
	}
	pc->next=pa?pa:pb; // 插入剩余段
	delete Lb; // 释放Lb 的头结点
}

(2)将两个非递减的有序链表合并为一个非递增的有序链表。要求结果链表仍使用原来两个链表的存储空间,不另外占用其他的存储空间。表中允许有重复的数据。

[ 题目分析]
合并后的新表使用头指针Lc 指向,pa 和pb 分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点,从第一个结点开始进行比较,当两个链表La 和Lb 均为到达表尾结点时,依次摘取其中较小者重新链接在Lc 表的表头结点之后,如果两个表中的元素相等,只摘取La 表中的元素,保留Lb 表中的元素。当一个表到达表尾结点,为空时,将非空表的剩余元素依次摘取,链接在Lc 表的表头结点之后。

[ 算法描述]

void MergeList(LinkList& La, LinkList& Lb, LinkList& Lc, )
{	// 合并链表La 和Lb,合并后的新表使用头指针Lc 指向
	pa=La->next;
	pb=Lb->next;
	//pa 和pb 分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点
	Lc=pc=La; 		// 用La 的头结点作为Lc 的头结点
	Lc->next=NULL;
	while(pa||pb ){			// 只要存在一个非空表,用q 指向待摘取的元素
		if(!pa) {			//La 表为空,用q 指向pb , pb指针后移
			q=pb; 
			pb=pb->next;
		}else if(!pb) {		//Lb表为空,用q指向pa,pa指针后移
			q=pa;
			pa=pa->next;
		}else if(pa->data<=pb->data) {		// 取较小者(包括相等) La 中的元素,用q指向pa,pa指针后移
			q=pa; 
			pa=pa->next;
		}else {					// 取较小者Lb 中的元素,用q 指向pb, pb 指针后移
			q=pb;
			pb=pb->next;
		}
		// 将q 指向的结点插在Lc 表的表头结点之后
		q->next = Lc->next;
		Lc->next = q;
	}
	delete Lb; 				// 释放Lb的头结点
}

(3)已知两个链表A和B分别表示两个集合,其元素递增排列。请设计一个算法,用于求出A与B的交集,并存放在A链表中。

[ 题目分析]
只有同时出现在两集合中的元素才出现在结果表中, 合并后的新表使用头指针Lc 指向。pa 和pb 分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点, 从第一个结点开始进行比较,当两个链表La 和Lb 均为到达表尾结点时,如果两个表中相等的元素时,摘取La 表中的元素,删除Lb 表中的元素;如果其中一个表中的元素较小时,删除此表中较小的元素,此表的工作指针后移。当链表La 和Lb 有一个到达表尾结点,为空时,依次删除另一个非空表中的所有元素。

[ 算法描述]

void Mix(LinkList& La, LinkList& Lb, LinkList& Lc)
{		//pa 和pb 分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点
	pa=La->next;
	pb=Lb->next;
	Lc=pc=La; 		// 用La 的头结点作为Lc 的头结点
	while(pa&&pb){ 
		if(pa->data==pb- >data){  	//交集并入结果表中。
			pc->next=pa;
			pc=pa;
			pa=pa->next;
			u=pb;
			pb=pb->next;
			delete u;
		}else if(pa->data<pb->data) {
			u=pa;
			pa=pa->next;
			delete u;
		}else {
			u=pb;
			pb=pb->next;
			delete u;
		}
	}
	while(pa){u=pa; pa=pa->next; delete u;}     //释放结点空间
	while(pb) {u=pb; pb=pb->next; delete u ;}   //释放结点空间
	pc->next=null;    	//置链表尾标记。
	delete Lb; 			// 释放Lb 的头结点
}

(4) 巳知两个链表A 和B 分别表示两个集合,其元素递增排列。请设计算法求出两个集合A和B 的差集(即仅由在A中出现而不在B中出现的元素所构成的集合),并以同样的形式存储,同时返回该集合的元素个数。

[ 题目分析]
求两个集合A 和B 的差集是指在A 中删除A 和B 中共有的元素,即删除链表中的相应结点, 所以要保存待删除结点的前驱,使用指针pre 指向前驱结点。pa 和pb 分别是链表La 和Lb 的工作指针, 初始化为相应链表的第一个结点,从第一个结点开始进行比较,当两个链表La 和Lb 均为到达表尾结点时,如果La 表中的元素小于Lb 表中的元素, pre 置为La 表的工作指针pa 删除Lb 表中的元素;如果其中一个表中的元素较小时,删除此表中较小的元素,此表的工作指针后移。当链表La 和Lb 有一个为空时, 依次删除另一个非空表中的所有元素。

[ 算法描述]

void Difference ( LinkList& La, LinkList& Lb,int *n )
{ 			//差集的结果存储于单链表La中, *n是结果集合中元素个数,调用时为0
	//pa 和pb 分别是链表La和Lb的工作指针, 初始化为相应链表的第一个结点
	pa=La->next; 
	pb=Lb->next;
	pre=La;			// pre为La中pa所指结点的前驱结点的指针
	while(pa&&pb){
		// A链表中当前结点指针后移
		if(pa->data<q->data){pre=pa;pa=pa->next;*n++;}
		//B 链表中当前结点指针后移
		else if(pa->data>q->data)q=q->next; 
		//处理A, B中元素值相同的结点,应删除
		//删除结点
		else {pre->next=pa->next; 
			u=pa; pa=pa->next;delete u;} 
	}
}

(5)设计算法将一个带头结点的单链表A分解为两个具有相同结构的链表B和C, 其中B表的结点为A 表中值小千零的结点,而C 表的结点为A 表中值大于零的结点(链表A 中的元素为非零整数,要求B 、C 表利用A 表的结点)。

[ 题目分析]
B表的头结点使用原来A表的头结点,为C表新申请一个头结点。从A表的第一个结点开始,依次取其每个结点p,判断结点p的值是否小于0,利用前插法,将小于0的结点插入B表, 大于等于0的结点插入C表。

[ 算法描述]

void DisCompose(LinkedList A)
{ 
	B=A;
	B->next= NULL; 			//B 表初始化
	C=new LNode;			//为C 申请结点空间
	C->next=NULL; 			// C 初始化为空表
	p=A->next; 				//p 为工作指针
	while(p!= NULL){
		r=p->next; 			//暂存p 的后继
		if(p->data<0){p->next=B->next; B- >next=p; } 		//将小于0 的结点链入B 表, 前插法
		else {p->next=C->next; C- >next=p; } 				//将大于等于0 的结点链入C 表, 前插法
		p=r; 												//p 指向新的待处理结点。
	}
}

(6)设计一个算法,通过一趟遍历确定长度为n 的单链表中值最大的结点。

[ 题目分析]
假定第一个结点中数据具有最大值,依次与下一个元素比较,若其小于下一个元素,则设其下一个元素为最大值,反复进行比较,直到遍历完该链表。

[ 算法描述]

ElemType Max (LinkList L ){
	if(L->next==NULL) return NULL;
	pmax=L->next; 								// 假定第一个结点中数据具有最大值
	p=L->next->next;
	while(p != NULL ){							// 如果下一个结点存在
		if(p->data > pmax->data) pmax=p;		// 如果p 的值大于pmax 的值,则重新赋值
		p=p->next;								// 遍历链表
	}
	return pmax->data;
}

(7)设计一个算法,将链表中所有结点的链接方向“原地” 逆转,即要求仅利用原表的存储空间,换句话说,要求算法的空间复杂度为0(1)。

[ 题目分析]
从首元结点开始,逐个地把链表L 的当前结点p 插入新的链表头部。

[ 算法描述]

void inverse(LinkList &L)
{				// 逆置带头结点的单链表 L
	p=L->next;
	L->next=NULL;
	while (p) {
		q=p->next; 			// q 指向*p 的后继
		p->next=L->next;
		L->next=p;			// *p 插入在头结点之后
		p = q;
	}
}

(8)设计一个算法,删除递增有序链表中值大千mink且小千maxk:的所有元素(mink和maxk是给定的两个参数,其值可以和表中的元素相同,也可以不同)。

[ 题目分析]
分别查找第一个值>mink 的结点和第一个值≥ maxk 的结点,再修改指针,删除值大于mink 且小于maxk 的所有元素。

[ 算法描述]

void delete(LinkList &L, int mink, int maxk)
{
	p=L->next; 					// 首元结点
	while (p && p->data<=mink){ pre=p; p=p->next; }			 // 查找第一个值>mink 的结点
	if (p){
		while (p && p->data<maxk) p=p->next;				// 查找第一个值≥ maxk 的结点
		q=pre->next; pre->next=p; 							// 修改指针
		while (q!=p){ s=q->next; delete q; q=s; } 			// 释放结点空间
	}
}

(9)巳知p指向双向循环链表中的一个结点,其结点结构为data 、prior、next三个域,写出算法change§, 交换p所指向的结点及其前驱结点的顺序。

[ 题目分析]
知道双向循环链表中的一个结点,与前驱交换涉及到四个结点( p 结点,前驱结点,前驱的前驱结点,后继结点)六条链。

[ 算法描述]

void Exchange ( LinkedList p )
{		// p 是双向循环链表中的一个结点,本算法将p 所指结点与其前驱结点交换。
	q=p->llink;
	q->llink->rlink=p;				//p 的前驱的前驱之后继为p
	p->llink=q->llink;				//p 的前驱指向其前驱的前驱。
	q->rlink=p->rlink;				//p 的前驱的后继为p 的后继。
	q->llink=p;						//p 与其前驱交换
	p->rlink->llink=q;				//p 的后继的前驱指向原p 的前驱
	p->rlink=q;						// p 的后继指向其原来的前驱
}

(10)已知长度为n的线性表A采用顺序存储结构,请写一个时间复杂度为O(n)、空间复杂度为0(1)的算法,该算法可删除线性表中所有值为item的数据元素。

[ 题目分析]
在顺序存储的线性表上删除元素,通常要涉及到一系列元素的移动(删第i 个元素,第 i+1 至第n 个元素要依次前移) 。本题要求删除线性表中所有值为item 的数据元素,并未要求元素间的相对位置不变。因此可以考虑设头尾两个指针( i=1 , j=n ),从两端向中间移动,凡遇到值item 的数据元素时,直接将右端元素左移至值为item 的数据元素位置。

[ 算法描述]

// A 是有 n 个元素的一维数组,本算法删除A 中所有值为 item 的元素。
void Delete ( ElemType A[ ], int n )
{	//设置数组低、高端指针(下标) 。
	i=1; j=n;			
	while(i<j){
		while(i<j && A[i]!=item)i++;		//若值不为item ,左移指针。
		if(i<j)	while(i<j && A[j]==item)  j--;	//若右端元素为item ,指针左移
		if(i<j) A[i++]=A[j--];
	}
}
  • 48
    点赞
  • 225
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值