(1)递归实现
1.对空链表存在的情况进行处理,假如pHead1为空则返回pHead2,pHead2为空则返回pHead1。(两个都为空此情况在pHead1为空已经被拦截)
2.比较两个链表第一个结点的大小确定头结点的位置
3.头结点确定后 继续在剩下的结点中 选出下一个结点去链接到第二步选出的结点后面,然后在继续重复2 3步 直到有链表为空
举个实例:有有序单链表L1 (1 3 5 7 ), L2 (2 4 6 8)
1.链表L1的第一个结点的值小于链表L2的第一个结点的值,因此链表1的头结点是合并后链表的头结点。
2.在剩下的结点中链表L2的第一个结点的值小于链表L1的第一个结点的值,因此将链表二的第一个结点与第一步的头结点连接。(就是每次选择两个链表头结点中最小结点添加到合并链表的后面)
3.然后继续在剩下的结点中重复第二、三步,直到有链表为空。
代码如下:
PNode MergeTwoOrderedLists(PNode pHead1, PNode pHead2)
{
PNode newHead = NULL;
if (NULL == pHead1)
{
return pHead2;
}
else if(NULL ==pHead2)
{
return pHead2;
}
else
{
if (pHead1->data < pHead2->data)
{
newHead = pHead1;
newHead->next = MergeTwoOrderedLists(pHead1->next, pHead2);
}
else
{
newHead = pHead2;
newHead->next = MergeTwoOrderedLists(pHead1, pHead2->next);
}
return newHead;
}
}
(2)非递归实现
1.第一步与递归一样,对空链表存在的情况进行处理,假如pHead1为空则返回pHead2,pHead2为空则返回pHead1。(两个都为空此情况在pHead1为空已经被拦截)
2.在两个链表无空链表的情况下确定第一个结点,比较链表1和链表2的第一个结点的值,将值小的结点保存下来为合并后的第一个结点。并且把第一个结点为最小的链表向后移动一个元素。
3.继续在剩下的元素中选择小的值,连接到第一个结点后面,并不断next将值小的结点连接到第一个结点后面,直到某一个链表为空。
4.当两个链表长度不一致时,也就是比较完成后其中一个链表为空,此时需要把另外一个链表剩下的元素都连接到第一个结点的后面。
代码如下:
PNode MergeTwoOrderedLists(PNode pHead1, PNode pHead2)
{
PNode pTail = NULL;//指向新链表的最后一个结点 pTail->next去连接
PNode newHead = NULL;//指向合并后链表第一个结点
if (NULL == pHead1)
{
return pHead2;
}
else if(NULL == pHead2)
{
return pHead1;
}
else
{
//确定头指针
if ( pHead1->data < pHead2->data)
{
newHead = pHead1;
pHead1 = pHead1->next; //指向链表的第二个结点
}
else
{
newHead = pHead2;
pHead2 = pHead2->next;
}
pTail = newHead; //指向第一个结点
while ( pHead1 && pHead2)
{
if ( pHead1->data <= pHead2->data )
{
pTail->next = pHead1;
pHead1 = pHead1->next;
}
else
{
pTail->next = pHead2;
pHead2 = pHead2->next;
}
pTail = pTail->next;
}
if(NULL == pHead1)
{
pTail->next = pHead2;
}
else if(NULL == pHead2)
{
pTail->next = pHead1;
}
return newHead;
}