查找两个单链表的公共节点
算法思路:首先要明白单链表有公共节点的情况是什么,因为单链表中每个节点的next指针是唯一的,所以当存在公共节点的时候,其后面的所有节点也都是相同的,所以这时的链表是一个“Y”型如图所示,而不是我们惯性思维想到的“X”型。根据这个结论可知,如果两个链表存在公共节点的话,它们的尾结点也肯定是相同的。
明白了有公共节点的链表构造之后就好办了,假设两个链表的长度分别是 len1 和 len2
第一种暴力方法:遍历第一个链表的节点,循环嵌套遍历第二个链表的所有节点,直到找到相同的节点。时间复杂度为O(len1*len2),这种方法实现比较简单,但是效率不高,这里就不放出来。
第二种优化的方法:两个链表自从公共节点开始,之后的长度都是相同的,所以只要遍历两个链表长度一样的地方,这样可以减少遍历的次数。
先得到两个链表的长度, longptr指向较长的链表,shortptr指向较短的链表,dist用来存储两个链表长度的差值,先将longptr后移dist个长度,这样能保证之后遍历的次数相等,然后就可以通过一次遍历比较完所有的节点。具体实现代码如下。
//查找两个链表的公共节点
LNode* Search_SamNode(LinkList L1,LinkList L2){
//获取两个链表的长度 dist表示长度差
int len1 = Length(L1),
len2 = Length(L2),
dist;
LNode* longptr,shortptr;
if(len1 > len2){
dist = len1 - len2;
longptr = L1;
shortptr = L2;
}else{
dist = len2 - len1;
longptr = L2;
shortptr = L1;
}
//将长指针后移dist,保证两个指针同步
while(dist--)
longptr = longptr->next;
while(longptr != NULL){
if(longptr == shortptr)
return longptr;
else{
longptr = longptr->next;
shortptr = shortptr->next;
}
}
return NULL;
}