思路:
1、排序
先将两个链表从小到大排序,调整成两个有序链表。
2、合并
使用两个工作指针PA和PB分别从两个表头开始循环。
当两个工作指针均非空时,如果PA->data大于PB->data,将PB插入到新链表中,PB指针向后移动;如果PA->data小于PB->data,将PA插入到新链表中,PA指针向后移动;如果PA->data等于PB->data,将PB(或PA)插入到新链表中,两个指针同时向后移动。
当一个指针为空时,证明有一个链表已经走到了最后。如果PA非空,将PA后的元素全部加入新链表中,如果PB非空,将PB后的元素全部加入新链表中。
代码实现:
#include <iostream>
#include <stdlib.h>
using namespace std;
//定义链表结点
typedef struct Node
{
int data;
Node *next;
}*LinkList;
//创建链表
void createLinkList(LinkList &l,int n)
{
l=(LinkList)malloc(sizeof(Node));
l->next=NULL;
Node *s;
for(int i=0;i<n;i++)
{
s=(Node*)malloc(sizeof(Node));
cin>>s->data;
s->next=l->next;
l->next=s;
}
}
//显示链表
void show(LinkList c)
{
Node *tmp;
tmp=c->next;
while(tmp)
{
cout<<tmp->data<<" ";
tmp=tmp->next;
}
cout<<endl;
}
//链表排序-直接选择排序
void sortList(LinkList l)
{
Node *s,*tmp,*t;
s=l->next;
int minum,n;
while(s)
{
minum=s->data;
t=s;
tmp=s->next;
while(tmp)
{
if(tmp->data<minum)
{
minum=tmp->data;
t=tmp;
}
tmp=tmp->next;
}
n=t->data;
t->data=s->data;
s->data=n;
s=s->next;
}
}
//合并链表
LinkList unionSet(LinkList p,LinkList q)
{
Node *pa,*pb,*pc;
LinkList r=(LinkList)malloc(sizeof(Node)); //申请结点
r->next=NULL; //初始化链表r
pa=p->next; //链表p的工作指针
pb=q->next; //链表q的工作指针
pc=r; //新链表r的工作指针
while(pa&&pb)
{
if(pa->data>pb->data) //q中元素插入r
{
pc->next=pb;
pc=pb;
pb=pb->next;
}
else //p中元素插入r
{
if(pa->data==pb->data) //若两个数据元素相等,工作指针同时后移
pb=pb->next;
pc->next=pa;
pc=pa;
pa=pa->next;
}
}
if(pa) pc->next=pa; //若a未到末尾,将c指向a
if(pb) pc->next=pb; //若b未到末尾,将c指向b
show(r);
}
int main()
{
LinkList p,q;
int length_p,length_q;
cin>>length_p;
createLinkList(p,length_p); //创建链表
sortList(p); //链表排序
cin>>length_q;
createLinkList(q,length_q);
sortList(q);
unionSet(p,q); //合并链表
return 0;
}