• [LeetCode]-160. 相交链表-141. 环形链表-142.环形链表II-138.随机链表的复制


    目录

    160.相交链表

     题目

    思路

    代码 

    141.环形链表

     题目

    思路

    代码

    142.环形链表II

    题目

    思路

    代码


    160.相交链表

    160. 相交链表 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/intersection-of-two-linked-lists/description/

     题目

    给你两个单链表的头节点 headA 和 headB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null 。

    图示两个链表在节点 c1 开始相交

    题目数据 保证 整个链式结构中不存在环。

    注意,函数返回结果后,链表必须 保持其原始结构 。

    示例:

    输入:intersectVal = 8, listA = [4,1,8,4,5], listB = [5,6,1,8,4,5], skipA = 2, skipB = 3
    输出:Intersected at '8'
    解释:相交节点的值为 8 (注意,如果两个链表相交则不能为 0)。
    从各自的表头开始算起,链表 A 为 [4,1,8,4,5],链表 B 为 [5,6,1,8,4,5]。
    在 A 中,相交节点前有 2 个节点;在 B 中,相交节点前有 3 个节点。
    — 请注意相交节点的值不为 1,因为在链表 A 和链表 B 之中值为 1 的节点 (A 中第二个节点和 B 中第三个节点) 是不同的节点。换句话说,它们在内存中指向两个不同的位置,而链表 A 和链表 B 中值为 8 的节点 (A 中第三个节点,B 中第四个节点) 在内存中指向相同的位置。

    思路

     先分别从两个单链表A、B的头节点开始遍历,分别计算出两个单链表的长度lenA、lenB,由于相交节点后面部分链表是共用的,所以长度差只存在于相交节点前面部分,因此,我们可以让较长的链表(可以先假设A长)先往后遍历差距步abs(lenA-lenB),再让A、B链表同时向后遍历(longlistA=longlistA->next、shortlistB=shortlistB->next),当A、B遍历到相同的节点时,即longlistA==shortlistB时,同时到达相交起始节点,返回longlist或者shortlist均为相交起始节点。

    图示👇

    代码 

    1. struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    2. struct ListNode* curA=headA;
    3. struct ListNode* curB=headB;
    4. int lenA=1,lenB=1;
    5. while(curA->next)
    6. {
    7. curA=curA->next;
    8. lenA++;
    9. }
    10. while(curB->next)
    11. {
    12. curB=curB->next;
    13. lenB++;
    14. }
    15. if(curA!=curB)
    16. return NULL;
    17. int grp=0;
    18. if(lenA>lenB)
    19. grp=lenA-lenB;
    20. else
    21. grp=lenB-lenA;
    22. struct ListNode* longlist=headA,* shortlist = headB;
    23. if(lenA
    24. {
    25. longlist=headB;
    26. shortlist=headA;
    27. }
    28. //长的先走差距步
    29. while(grp--)
    30. {
    31. longlist=longlist->next;
    32. }
    33. while(longlist!=shortlist)
    34. {
    35. longlist=longlist->next;
    36. shortlist=shortlist->next;
    37. }
    38. return longlist;
    39. }

    141.环形链表

    141. 环形链表 - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/linked-list-cycle/description/

     题目

    给你一个链表的头节点 head ,判断链表中是否有环。

    如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos 不作为参数进行传递 。仅仅是为了标识链表的实际情况。

    如果链表中存在环 ,则返回 true 。 否则,返回 false 。

    示例:

    输入:head = [3,2,0,-4], pos = 1
    输出:true
    解释:链表中有一个环,其尾部连接到第二个节点。

    思路

    利用快慢指针法,在头节点创建两个指针fast和slow,从头节点向后分别开始遍历,但是fast一次走两步(fast=fast->next->next),slow一次走一步(slow=slow->next),如果链表中有环,没有指向NULL的节点,fast和slow进入环后会无限循环遍历下去,因为fast遍历速度是slow的两倍,在环中,fast和slow总会遍历到同一节点,我们就可以添加一个跳出条件,当fast和slow相等时(fast==slow),证明链表中有环,返回true。

    图示👇

    代码

    1. bool hasCycle(struct ListNode *head) {
    2. struct ListNode* slow=head,* fast=head;
    3. while(fast&&fast->next)
    4. {
    5. slow=slow->next;
    6. fast=fast->next->next;
    7. if(fast==slow)
    8. return true;
    9. }
    10. return false;
    11. }

    142.环形链表II

     142. 环形链表 II - 力扣(LeetCode)icon-default.png?t=N7T8https://leetcode.cn/problems/linked-list-cycle-ii/description/

    题目

    给定一个链表的头节点  head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null

    如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。如果 pos 是 -1,则在该链表中没有环。注意:pos 不作为参数进行传递,仅仅是为了标识链表的实际情况。

    不允许修改 链表。

    示例:

    输入:head = [3,2,0,-4], pos = 1
    输出:返回索引为 1 的链表节点
    解释:链表中有一个环,其尾部连接到第二个节点。

    思路

    遇到环形链表问题,我们可以先考虑用快慢指针求出相遇点,在这道题中,我们要求出链表入环的第一个节点,我们知道链表成环需要尾节点指向链表中的任意节点,被指向节点就有了两个指向它的节点,我们可以利用求相交节点的方法求入环的第一个节点,类似于上面相交链表题,为了创造条件,我们可以断开相遇点,创建一个新节点newhead,和原节点head,两个节点带入上面求相交节点的题目中,就能求出入环第一个节点了。

    代码

    1. //求相交节点的函数
    2. struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {
    3. struct ListNode* curA=headA;
    4. struct ListNode* curB=headB;
    5. int lenA=1,lenB=1;
    6. while(curA->next)
    7. {
    8. curA=curA->next;
    9. lenA++;
    10. }
    11. while(curB->next)
    12. {
    13. curB=curB->next;
    14. lenB++;
    15. }
    16. if(curA!=curB)
    17. return NULL;
    18. int grp=0;
    19. if(lenA>lenB)
    20. grp=lenA-lenB;
    21. else
    22. grp=lenB-lenA;
    23. struct ListNode* longlist=headA,* shortlist = headB;
    24. if(lenA
    25. {
    26. longlist=headB;
    27. shortlist=headA;
    28. }
    29. //长的先走差距步
    30. while(grp--)
    31. {
    32. longlist=longlist->next;
    33. }
    34. while(longlist!=shortlist)
    35. {
    36. longlist=longlist->next;
    37. shortlist=shortlist->next;
    38. }
    39. return longlist;
    40. }
    41. struct ListNode *detectCycle(struct ListNode *head) {
    42. struct ListNode* fast,* slow;
    43. fast=slow=head;
    44. while(fast&&fast->next)
    45. {
    46. fast=fast->next->next;
    47. slow=slow->next;
    48. if(fast==slow)
    49. {
    50. struct ListNode* meet=fast;
    51. //断开相遇点
    52. struct ListNode* newhead=meet->next;
    53. meet->next=NULL;
    54. //找相交节点
    55. return getIntersectionNode(newhead,head);
    56. }
    57. }
    58. return NULL;
    59. }
  • 相关阅读:
    用Postman发送xml数据
    总结《你不知道的JavaScript》三卷小记
    高校校园网规划与设计
    浅浅的 使用网络调试助手和Mosquitto 分析MQTT协议数据包内容
    2. 慢查询、索引、执行计划详解
    计算机毕业设计Java交通事故档案管理系统(源码+mysql数据库+系统+lw文档)
    《Mission Red in Slience》——这本“连环画”值得一看
    SpringFramework 之EnableAsync
    移动WEB开发之rem布局--苏宁首页案例制作(技术方案1)
    LeetCode 每日一题 2022/8/29-2022/9/4
  • 原文地址:https://blog.csdn.net/m0_64476561/article/details/134232936