for(j=1;j<=n-i;j++)
x++;
(6)x=n;//n>1
y=0;
while(x≥(y+1)*(y+1))
y++;
(1)O
(1)
(2)O(m*n)
(3)O(n2)
(4)O(log3n)
(5)因为x++共执行了n-1+n-2+……+1=n(n-1)/2,所以执行时间为O(n2)
(6)O(
)
第2章线性表
1.选择题
(1)一个向量第一个元素的存储地址是100,每个元素的长度为2,则第5个元素的地址是()。
A.110B.108C.100D.120
(2)在n个结点的顺序表中,算法的时间复杂度是O
(1)的操作是()。
A.访问第i个结点(1≤i≤n)和求第i个结点的直接前驱(2≤i≤n)
B.在第i个结点后插入一个新结点(1≤i≤n)
C.删除第i个结点(1≤i≤n)
D.将n个结点从小到大排序
(3)向一个有127个元素的顺序表中插入一个新元素并保持原来顺序不变,平均要移动的元素个数为()。
A.8B.63.5C.63D.7
(4)链接存储的存储结构所占存储空间()。
A.分两部分,一部分存放结点值,另一部分存放表示结点间关系的指针
B.只有一部分,存放结点值
C.只有一部分,存储表示结点间关系的指针
D.分两部分,一部分存放结点值,另一部分存放结点所占单元数
(5)线性表若采用链式存储结构时,要求内存中可用存储单元的地址()。
A.必须是连续的B.部分地址必须是连续的
C.一定是不连续的D.连续或不连续都可以
(6)线性表L在()情况下适用于使用链式结构实现。
A.需经常修改L中的结点值B.需不断对L进行删除插入
C.L中含有大量的结点D.L中结点结构复杂
(7)单链表的存储密度()。
A.大于1B.等于1C.小于1D.不能确定
(8)将两个各有n个元素的有序表归并成一个有序表,其最少的比较次数是()。
A.nB.2n-1C.2nD.n-1
(9)在一个长度为n的顺序表中,在第i个元素(1≤i≤n+1)之前插入一个新元素时须向后移动()个元素。
A.n-iB.n-i+1C.n-i-1D.i
(10)线性表L=(a1,a2,……an),下列说法正确的是()。
A.每个元素都有一个直接前驱和一个直接后继
B.线性表中至少有一个元素
C.表中诸元素的排列必须是由小到大或由大到小
D.除第一个和最后一个元素外,其余每个元素都有一个且仅有一个直接前驱和直接后继。
(11)若指定有n个元素的向量,则建立一个有序单链表的时间复杂性的量级是()。
A.O
(1)B.O(n)C.O(n2)D.O(nlog2n)
(12)以下说法错误的是()。
A.求表长、定位这两种运算在采用顺序存储结构时实现的效率不比采用链式存储结构时实现的效率低
B.顺序存储的线性表可以随机存取
C.由于顺序存储要求连续的存储区域,所以在存储管理上不够灵活
D.线性表的链式存储结构优于顺序存储结构
(13)在单链表中,要将s所指结点插入到p所指结点之后,其语句应为()。
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;
(14)在双向链表存储结构中,删除p所指的结点时须修改指针()。
A.p->next->prior=p->prior;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;
(15)在双向循环链表中,在p指针所指的结点后插入q所指向的新结点,其修改指针的操作是()。
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;
2.算法设计题
(1)将两个递增的有序链表合并为一个递增的有序链表。
要求结果链表仍使用原来两个链表的存储空间,不另外占用其它的存储空间。
表中不允许有重复的数据。
voidMergeList_L(LinkList&La,LinkList&Lb,LinkList&Lc){
pa=La->next;pb=Lb->next;
Lc=pc=La;//用La的头结点作为Lc的头结点
while(pa&&pb){
if(pa->datadata){pc->next=pa;pc=pa;pa=pa->next;}
elseif(pa->data>pb->data){pc->next=pb;pc=pb;pb=pb->next;}
else{//相等时取La的元素,删除Lb的元素
pc->next=pa;pc=pa;pa=pa->next;
q=pb->next;deletepb;pb=q;}
}
pc->next=pa?
pa:
pb;//插入剩余段
deleteLb;//释放Lb的头结点}
(2)将两个非递减的有序链表合并为一个非递增的有序链表。
要求结果链表仍使用原来两个链表的存储空间,不另外占用其它的存储空间。
表中允许有重复的数据。
voidunion(LinkList&La,LinkList&Lb,LinkList&Lc,){
pa=La->next;pb=Lb->next;//初始化
Lc=pc=La;//用La的头结点作为Lc的头结点
Lc->next=NULL;
while(pa||pb){
if(!
pa){q=pb;pb=pb->next;}
elseif(!
pb){q=pa;pa=pa->next;}
elseif(pa->data<=pb->data){q=pa;pa=pa->next;}
else{q=pb;pb=pb->next;}
q->next=Lc->next;Lc->next=q;//插入
}
deleteLb;//释放Lb的头结点}
(3)已知两个链表A和B分别表示两个集合,其元素递增排列。
请设计算法求出A与B的交集,并存放于A链表中。
voidMix(LinkList&La,LinkList&Lb,LinkList&Lc,){
pa=la->next;pb=lb->next;∥设工作指针pa和pb;
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;deleteu;}
elseif(pa->datadata){u=pa;pa=pa->next;deleteu;}
else{u=pb;pb=pb->next;deleteu;}
while(pa){u=pa;pa=pa->next;deleteu;}∥释放结点空间
while(pb){u=pb;pb=pb->next;deleteu;}∥释放结点空间
pc->next=null;∥置链表尾标记。
deleteLb;∥注:
本算法中也可对B表不作释放空间的处理
(4)已知两个链表A和B分别表示两个集合,其元素递增排列。
请设计算法求出两个集合A和B的差集(即仅由在A中出现而不在B中出现的元素所构成的集合),并以同样的形式存储,同时返回该集合的元素个数。
voidDifference(LinkedListA,B,*n)
∥A和B均是带头结点的递增有序的单链表,分别存储了一个集合,本算法求两集合的差集,存储于单链表A中,*n是结果集合中元素个数,调用时为0
{p=A->next;∥p和q分别是链表A和B的工作指针。
q=B->next;pre=A;∥pre为A中p所指结点的前驱结点的指针。
while(p!
=null&&q!
=null)
if(p->datadata){pre=p;p=p->next;*n++;}∥A链表中当前结点指针后移。
elseif(p->data>q->data)q=q->next;∥B链表中当前结点指针后移。
else{pre->next=p->next;∥处理A,B中元素值相同的结点,应删除。
u=p;p=p->next;deleteu;}∥删除结点
(5)设计算法将一个带头结点的单链表A分解为两个具有相同结构的链表B、C,其中B表的结点为A表中值小于零的结点,而C表的结点为A表中值大于零的结点(链表A的元素类型为整型,要求B、C表利用A表的结点)。
(6)设计一个算法,通过一趟遍历在单链表中确定值最大的结点。
ElemTypeMax(LinkListL){
if(L->next==NULL)returnNULL;
pmax=L->next;//假定第一个结点中数据具有最大值
p=L->next->next;
while(p!
=NULL){//如果下一个结点存在
if(p->data>pmax->data)pmax=p;
p=p->next;
}
returnpmax->data;
(7)设计一个算法,通过遍历一趟,将链表中所有结点的链接方向逆转,仍利用原表的存储空间。
voidinverse(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是给定的两个参数,其值可以和表中的元素相同,也可以不同)。
voiddelete(LinkList&L,intmink,intmaxk){
p=L->next;//首元结点
while(p&&p->data<=mink)
{pre=p;p=p->next;}//查找第一个值>mink的结点
if(p){
while(p&&p->datanext;
//查找第一个值≥maxk的结点
q=pre->next;pre->next=p;//修改指针
while(q!
=p)
{s=q->next;deleteq;q=s;}//释放结点空间
}//if
}
(9)已知p指向双向循环链表中的一个结点,其结点结构为data、prior、next三个域,写出算法change(p),交换p所指向的结点和它的前缀结点的顺序。
知道双向循环链表中的一个结点,与前驱交换涉及到四个结点(p结点,前驱结点,前驱的前驱结点,后继结点)六条链。
voidExchange(LinkedListp)
∥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的后继指向其原来的前驱
}∥算法exchange结束。
(10)已知长度为n的线性表A采用顺序存储结构,请写一时间复杂度为O(n)、空间复杂度为O
(1)的算法,该算法删除线性表中所有值为item的数据元素。
[题目分析]在顺序存储的线性表上删除元素,通常要涉及到一系列元素的移动(删第i个元素,第i+1至第n个元素要依次前移)。
本题要求删除线性表中所有值为item的数据元素,并未要求元素间的相对位置不变。
因此可以考虑设头尾两个指针(i=1,j=n),从两端向中间移动,凡遇到值item的数据元素时,直接将右端元素左移至值为item的数据元素位置。
voidDelete(ElemTypeA[],intn)
∥A是有n个元素的一维数组,本算法删除A中所有值为item的元素。
{i=1;j=n;∥设置数组低、高端指针(下标)。
while(i{while(i=item)i++;∥若值不为item,左移指针。
if(iif(i}
[算法讨论]因元素只扫描一趟,算法时间复杂度为O(n)。
删除元素未使用其它辅助空间,最后线性表中的元素个数是j。