142. Linked List Cycle II
题目:
Given a linked list, return the node where the cycle begins. If there is no cycle, return null
.
Note: Do not modify the linked list.
Follow up:
Can you solve it without using extra space?
给定一个链表的头指针,问你能不能只用常数的空间快速判断一个链表是不是有环,如果有环,返回环的起始位置。
代码:
不能用额外的空间,所以必须通过遍历,判断有没有环的存在。
经典的判断方法(百度很多哦):
对于判断链表是否有环,方法很简单,用两个指针,一开始都指向头结点,一个是快指针,一次走两步,一个是慢指针,一次只走一步,当两个指针重合时表示存在环了。
根据此方法:
如图,a代代表链表起始位置到环开始位置的距离;x代表环起始位置到快慢指针节点相遇点的距离;c代表环的长度。
假设慢指针和快指针在9处第一次相遇,于是又慢指针走的距离 s_slow=a+x,
快指针会多走一圈之后来追上慢指针,于是走的距离应该是 s_fast=a+n*c+x(快指针可能走了n圈)
因为快指针的速度是慢指针的2倍,所以距离s_fast=2*s_slow.于是有:a+x+nc = 2*(a+x) =>nc = a + x => a = (n-1)c+c-x
当慢指针走距离a时候,快指针走了n-1圈加c-x的距离。
所以如果把慢指针放回起点S,让他重启走完a距离,快指针从相遇点9继续走,他们第二次肯定会相遇在4,也就是环的起点。
因为满指针走完a的时间里,快指针刚好走完了整数圈加c-x的距离,肯定会停在圈的起点。
于是可以根据这个规律,求出环的起点:
java代码,不复杂,就是分别循环到两次,找到第二次相遇的点:
public ListNode detectCycle(ListNode head) {
if(head==null){return null;}
ListNode Slower = head;
ListNode Faster = head;
//循环到第一次相遇为止
while(Faster!=null && Slower != null){
Slower = Slower.next;
Faster = Faster.next;
if(Faster!=null){
Faster = Faster.next;
}
else{
return null;
}
System.out.println("Faster: "+Faster.val+" "+"Slower: "+Slower.val);
if(Slower==Faster){
Slower = head;
//循环到第二次相遇
while(Slower != Faster){
Slower = Slower.next;
Faster = Faster.next;
System.out.println("LittleCycle: Faster: "+Faster.val+" "+"Slower: "+Slower.val);
}
return Slower;
}
}
return null;
}
输出输出结果,构建如上图所示的环:
FirstCycle: Faster: 3 Slower: 2
FirstCycle: Faster: 5 Slower: 3
FirstCycle: Faster: 7 Slower: 4
FirstCycle: Faster: 9 Slower: 5
FirstCycle: Faster: 11 Slower: 6
FirstCycle: Faster: 5 Slower: 7
FirstCycle: Faster: 7 Slower: 8
FirstCycle: Faster: 9 Slower: 9
SecondCycle: Faster: 10 Slower: 2
SecondCycle: Faster: 11 Slower: 3
SecondCycle: Faster: 4 Slower: 4
环的起点: 4
问题代码逻辑不复复杂,但是解题思路很关键,也是百度学习了很久,当然测试也需要构建存在环的链表这种数据结构。
简单写下我的链表创建方法,也是凑了好久凑出来的,还在寻找好的方案:
先递归创建一个链表,然后把最后一个对象指向其中的一个(cycle=4),构建成环:)
本想直接递归创建一个带环的列表,但是很难在ListNode类中实现(至少我还没想出来:(),先这样用吧:
public class ListNode {
int val;
ListNode next;
ListNode(int x) {
val = x;
next = null;
}
//递归调用makeListNode,创建列表。
ListNode head;
public ListNode(int[] array){
head=makeListNode(array,0);
}
//根据对象的值获取链表中某个对象
public ListNode getNode(int value){
ListNode temp = this.head;
while(temp.val != value){
temp = temp.next;
}
return temp;
}
/**
* 采用递归的方式创建链表
* 传入的是链表的数组表示法
* 构造后是链表的链表表示法
*/
public static ListNode makeListNode(int[] array,int index){
if(index < array.length){
int value=array[index];
ListNode t=new ListNode(value);
t.next=makeListNode(array,++index);
return t;
}
return null;
}
}
public static void main(String[] args) {
// TODO Auto-generated method stub
Solution a = new Solution();
int[] arr = {1,2,3,4,5,6,7,8,9,10,11};
int cycle = 4;
ListNode list = new ListNode(arr);
//把链表最后一位指向链表中第cycle个对象
if(cycle>0){
for (int i = 0;i<arr.length;i++){
if(cycle==arr[i]){
list.getNode(arr[arr.length-1]).next = list.getNode(cycle);
}
}
}
ListNode x = a.detectCycle(list.head);
if(x==null){
System.out.println("yes");
}
else{
System.out.println(x.val);
}
最后,提交结果:
142. Linked List Cycle II的更多相关文章
- 141. Linked List Cycle&142. Linked List Cycle II(剑指Offer-链表中环的入口节点)
题目: 141.Given a linked list, determine if it has a cycle in it. 142.Given a linked list, return the ...
- leetcode 141. Linked List Cycle 、 142. Linked List Cycle II
判断链表有环,环的入口结点,环的长度 1.判断有环: 快慢指针,一个移动一次,一个移动两次 2.环的入口结点: 相遇的结点不一定是入口节点,所以y表示入口节点到相遇节点的距离 n是环的个数 w + n ...
- 【算法分析】如何理解快慢指针?判断linked list中是否有环、找到环的起始节点位置。以Leetcode 141. Linked List Cycle, 142. Linked List Cycle II 为例Python实现
引入 快慢指针经常用于链表(linked list)中环(Cycle)相关的问题.LeetCode中对应题目分别是: 141. Linked List Cycle 判断linked list中是否有环 ...
- 142. Linked List Cycle II【easy】
142. Linked List Cycle II[easy] Given a linked list, return the node where the cycle begins. If ther ...
- Java for LeetCode 142 Linked List Cycle II
Given a linked list, return the node where the cycle begins. If there is no cycle, return null. Foll ...
- 【LeetCode】142. Linked List Cycle II (2 solutions)
Linked List Cycle II Given a linked list, return the node where the cycle begins. If there is no cyc ...
- [LeetCode] 142. Linked List Cycle II 单链表中的环之二
Given a linked list, return the node where the cycle begins. If there is no cycle, return null. To r ...
- (链表 双指针) leetcode 142. Linked List Cycle II
Given a linked list, return the node where the cycle begins. If there is no cycle, return null. To r ...
- 【LeetCode】142. Linked List Cycle II
Difficulty:medium More:[目录]LeetCode Java实现 Description Given a linked list, return the node where t ...
- [LeetCode] 142. Linked List Cycle II 链表中的环 II
Given a linked list, return the node where the cycle begins. If there is no cycle, return null. Foll ...
随机推荐
- STM32F103xx bxCAN(Basic Extended CAN) 滤波机制
一.背景 最近一个项目需要使用STM32F103xx实现CAN通信,而CAN总线的消息滤波在各个MCU上有不同机制, 譬如,SJA1000为标识符位屏蔽滤波机制,NXP的LPC17xx系列为标识符列表 ...
- Cotex-M3内核LPC17xx系列时钟及其配置方法
一.背景: 最近正在接手一个项目,核心芯片既是LPC17XX系列MCU,内核为ARM的Cotex-M3内核. 想要玩转一个MCU,就一定得搞定其时钟! 时钟对MCU而言,就好比人类的心脏.由其给AHB ...
- 2.交通聚类 -层次聚类(agnes)Java实现
1.项目背景 在做交通路线分析的时候,客户需要找出车辆的行车规律,我们将车辆每天的行车路线当做一个数据样本,总共有365天或是更多,从这些数据中通过聚类来获得行车路线规律统计分析. 我首先想到是K-m ...
- lianjie
数值策划入门:如何确定游戏中的资源价值和定价http://bbs.gameres.com/thread_494366.html 一张常规的RPG游戏地图的制作流程 http://bbs.gameres ...
- c#中两种不同的存储过程调用与比较
存储过程简介 简单的说,存储过程是由一些SQL语句和控制语句组成的被封装起来的过程,它驻留在数据库中,可以被客户应用程序调用,也可以从另一个过程或触发器调用.它的参数可以被传递和返回.与应用程序中的函 ...
- jQuery Validate 表单验证
在做网页表单时时常需要在客户端对表单填写的数据进行验证一番才能提交,我们可以通过自己编写JavasScript代码来验证,但是有时数据量过多时就会有些难度了.基于jQuery的jquery.valid ...
- mysql基础面试
php面试题之五--MySQL数据库(基础部分) 五.MySQL数据库 mysql_num_rows() mysql_affected_rows() 这两个函数都作用于 mysql_query($qu ...
- phpcms学习的一点心得
最近,在学习phpcms的二次开发,因为要调试搜索模块程序,需建立若干栏目,按照栏目搜索.这一过程中涉及到phpcms的一个概念:模型. 以前并不太注意这个东东. 经过摸索,发现模型其实也是一个栏目的 ...
- BZOJ4411——[Usaco2016 Feb]Load balancing
1.题意: 给出N个平面上的点.保证每一个点的坐标都是正奇数. 你要在平面上画两条线,一条是x=a,一条是y=b,且a和b都是偶数. 直线将平面划成4个部分,要求包含点数最多的那个部分点数最少. 2. ...
- [BZOJ1901]Zju2112 Dynamic Rankings
[BZOJ1901]Zju2112 Dynamic Rankings 试题描述 给定一个含有n个数的序列a[1],a[2],a[3]--a[n],程序必须回答这样的询问:对于给定的i,j,k,在a[i ...