1. 题目:设带头结点的线性单链表A={a1,a2,...,am},B={b1,b2,...,bn} 。试编写算法按下列规则合并A、B为线性单链表C,使得
C={a1,b1,a2,b2,...am,bm,...,bn} , m<=n
或者
C={b1,a1,b2,a2,...,bn,an,...,am} , m>n
函数接口定义:
LinkList CombineList(LinkList La,LinkList Lb);
其中 La
和 Lb
都是用户传入的参数,分别为待合并单链表的头指针。函数须返回合并后的单链表的头指针。
裁判测试程序样例:
#include <stdio.h>
#include <stdlib.h>
typedef int DataType;
typedef struct node
{
DataType data;
struct node *next;
}LNode,*LinkList;
LinkList CreatLinkList(); //创建带头结点单链表,并返回头指针。
void PrintLinkList(LinkList H);//依次输出单链表H中各个元素结点,若为空表则输出NONE。
LinkList CombineList(LinkList La,LinkList Lb);
main()
{
LinkList la,lb;
la = CreatLinkList();
lb = CreatLinkList();
la=CombineList(la,lb);
PrintLinkList(la);
}
LinkList CreatLinkList()
{
int n,i;
LNode *nw,*rear=NULL,*head=NULL;
head=(LNode*)malloc(sizeof(LNode));
rear=head;
scanf("%d",&n);//接收结点总数
for(i=0;i<n;i++)
{
nw=(LNode*)malloc(sizeof(LNode));
scanf("%d",&nw->data);
rear->next=nw;
rear=nw;
}
rear->next=NULL;
return head;
}
void PrintLinkList(LinkList H)
{
LNode *p;
if(!(H->next))
{
printf("NONE\n");
return;
}
for(p=H->next;p;p=p->next)
printf("%d ",p->data);
printf("\n");
}
/* 请在这里填写答案 */
输入样例:
在这里给出一组输入。例如:
4
1 2 3 4
6
11 22 33 44 55 66
输出样例:
在这里给出相应的输出。例如:
1 11 2 22 3 33 4 44 55 66
2.注意事项:
①元素需交叉排列
②链表长度不一而导致需要元素剩余处理
3.答案与解析
LinkList CombineList(LinkList La,LinkList Lb){
int i,j;
LinkList p,q,r;
p=La->next;
q=Lb->next;
r=La;
while(p){ //La的长度
p=p->next;
i++;
}
while(q){ //Lb的长度
q=q->next;
j++;
}
p=La->next;
q=Lb->next;
if(i<=j){ //La短,从La首元素开始
while(p&&q){
r->next=p;
r=p;
p=p->next;
r->next=q;
r=q;
q=q->next;
r->next=q; //La中元素执行完毕,剩下皆为Lb中剩余元素
}else if(i>j){ //Lb短,从Lb首元素开始
while(p&&q){
r->next=q;
r=q;
q=q->next;
r->next=p;
r=p;
p=p->next;
}
r->next=p; //Lb中元素执行完毕,剩下皆为La中剩余元素
}
free(Lb);
return La;
}
4.分析总结
因为题目要求两链表交叉合并,故需根据链表长度不同分情况讨论,并且每次循环都要判断当前节点的 next 域是否为空