判断两个链表是否相交并找出交点

判断两个链表是否相交并找出交点

转载  2016年03月27日 10:01:27
  • 8369

问题描述:

一个比较经典的问题,判断两个链表是否相交,如果相交找出他们的交点。

第一种情况:两个链表均不含有环

image

思路:

1、直接法

采用暴力的方法,遍历两个链表,判断第一个链表的每个结点是否在第二个链表中,时间复杂度为O(len1*len2),耗时很大。

2、hash计数法

如 果 两个链表相交,则两个链表就会有共同的结点;而结点地址又是结点唯一标识。因而判断两个链表中是否存在地址一致的节点,就可以知道是否相交了。可以对第一 个链表的节点地址进行hash排序,建立hash表,然后针对第二个链表的每个节点的地址查询hash表,如果它在hash表中出现,则说明两个链表有共 同的结点。这个方法的时间复杂度为:O(max(len1+len2);但同时还得增加O(len1)的存储空间存储哈希表。这样减少了时间复杂度,增加 了存储空间。

以链表节点地址为值,遍历第一个链表,使用Hash保存所有节点地址值,结束条件为到最后一个节点(无环)或Hash中该地址值已经存在(有环)。

再遍历第二个链表,判断节点地址值是否已经存在于上面创建的Hash表中。

这个方面可以解决题目中的所有情况,时间复杂度为O(m+n)mn分别是两个链表中节点数量。由于节点地址指针就是一个整型,假设链表都是在堆中动态创建的,可以使用堆的起始地址作为偏移量,以地址减去这个偏移量作为Hash函数

3、第三种思路是比较奇特的,在编程之美上看到的。先遍历第一个链表到他的尾部,然后将尾部的next指针指向第二个链表(尾部指针的next本来指向的是null)。这样两个链表就合成了一个链表,判断原来的两个链表是否相交也就转变成了判断新的链表是否有环的问题了:即判断单链表是否有环?

这样进行转换后就可以从链表头部进行判断了,其实并不用。通过简单的了解我们就很容易知道,如果新链表是有环的,那么原来第二个链表的头部一定在环上。因此我们就可以从第二个链表的头部进行遍历的,从而减少了时间复杂度(减少的时间复杂度是第一个链表的长度)。

下图是一个简单的演示:

image

这种方法可以判断两个链表是否相交,但不太容易找出他们的交点。

4、仔细研究两个链表,如果他们相交的话,那么他们最后的一个节点一定是相同的,否则是不相交的。因此判断两个链表是否相交就很简单了,分别遍历到两个链表的尾部,然后判断他们是否相同,如果相同,则相交;否则不相交。示意图如下:

image

判断出两个链表相交后就是判断他们的交点了。假设第一个链表长度为len1,第二个问len2,然后找出长度较长的,让长度较长的链表指针向后移动|len1 - len2| (len1-len2的绝对值),然后在开始遍历两个链表,判断节点是否相同即可。

下面给出一个简单的实现:

/************************************************************************
   两个不含环的单链表的相交
   相交指的是结点的地址相同,而不是结点的值相同                    
************************************************************************/

[cpp]  view plain  copy
  1. typedef struct node_t  
  2. {  
  3. int data;//data  
  4. struct node_t *next; //next  
  5. }node;  
  6.    
  7. node* find_node(node *head1, node *head2)  
  8. {  
  9. if(NULL == head1 || NULL == head2)  
  10. {  
  11. return NULL;//如果有为空的链表,肯定是不相交的  
  12. }  
  13. node *p1, *p2;  
  14. p1 = head1;  
  15. p2 = head2;  
  16. int len1 = 0;  
  17. int len2 =0;  
  18. int diff = 0;  
  19. while(NULL != p1->next)  
  20. {  
  21. p1 = p1->next;  
  22. len1++;  
  23. }  
  24. while(NULL != p2->next)  
  25. {  
  26. p2 = p2->next;  
  27. len2++;  
  28. }  
  29. if(p1 != p2) //如果最后一个节点不相同,返回NULL  
  30. {  
  31. return NULL;  
  32. }  
  33. diff = abs(len1 - len2);  
  34. if(len1 > len2)  
  35. {  
  36. p1 = head1;  
  37. p2 = head2;  
  38. }  
  39. else  
  40. {  
  41. p1 = head2;  
  42. p2 = head1;  
  43. }  
  44. for(int i=0; i<diff; i++)  
  45. {  
  46. p1 = p1->next;  
  47. }  
  48. while(p1 != p2)  
  49. {  
  50. p1 = p1->next;  
  51. p2 = p2->next;  
  52. }  
  53. return p1;  
  54. }  

通过上面的操作就可以找到两个链表的交点了。

5、总结

上面的几种方法中最后一种是比较不错的,当然hash也是可以的。

问题的延伸:

如果原来的两个链表中有环怎么处理?

二、链表中有环时

(1)当两个链表中有环时,相交的判断:

如果链表有环且相交,那么这两个链表都是有环的。

找到第一个链表的环点,然后将环断开(当然不要忘记了保存它的下一个节点),然后再来遍历第二个链表,如果发现第二个链表从有环变成了无环,那么他们就是相交的嘛,否则就是不相交的了。

当两个有环的链表相交时,有以下两种情况:

在这种情况下,两个链表的交点在环点之前,可以将环点切断,这样就变成了两个无环的链表求相交点。可使用以上方法。

另一种情况为:

 

在这种情况下,不存在所谓的相交点。


(2)当一个链表中有环,一个链表中没有环时,两个链表必不相交。

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值