环问题 之 快慢指针

应用题目:力扣457

什么是快慢指针

双指针法的一种,双指针除了有左右双指针外还有快慢双指针
快慢指针是解决检测环问题的经典算法。
两个指针从同一个起点出发,快指针跑的更快,慢指针更慢。若是路径上存在环,那么快指针在进入环后会一直在环内打转,等待慢指针也进入后就能存在快指针追上慢指针的情况。若是路径上不存在环,那么快指针永远不会追上慢指针。

解题代码示例

class Solution {
    private int[] nums;
    private int n;
    public boolean circularArrayLoop(int[] _nums) {
        n = _nums.length;
        nums = _nums;
        for (int i = 0; i < n; i++) {
            int slow = i,fast = next(i);
            //保证同向查找,且不为0
            while(nums[slow]*nums[next(slow)]>0 && nums[fast]*nums[next(fast)]>0){
                if(slow==fast){
                    if(slow!=next(slow)){
                        return true;
                    }else{
                        break;
                    }
                }
                slow = next(slow);
                fast = next(next(fast));
            }
            int flag = i;
            //提高时间效率,把不可能的路径统统标记为0
            while(nums[flag]*nums[next(flag)]>0){
                int tmp = flag;
                flag = next(flag);
                nums[tmp] = 0;
            }
        }
        return false;
    }

    int next(int k){
        int res = (k+nums[k])%n;
        return res>=0?res:n+res;
    }
}

总结环问题解法

  1. 快慢指针:检测环存在
  2. 拓扑排序:检测环存在
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Java中,使用快慢指针(也称为龟兔赛跑法)是常见的判断链表是否有的方法。这种方法基于两个指针,一个每次移动一个节点,另一个每次移动两个节点。如果链表中有,那么快指针最终会追上慢指针;如果没有,快指针会先到达链表尾部。 下面是基本的实现步骤: 1. 初始化两个指针:`slow`(慢指针)和`fast`(快指针),分别指向链表的头节点。 2. 指针遍历:如果链表不为空,循执行以下操作: a. `slow`向前移动一步(`slow.next`)。 b. `fast`向前移动两步(`fast.next.next`)。 3. 判断的存在:如果`fast`指针在某次迭代中到达了`null`,说明链表没有,因为快指针走过的距离是慢指针的两倍,如果链表长度为偶数,快指针应该在链表末尾找到慢指针,如果奇数,则会先到尾部再回,不会追上。如果`fast`始终不为`null`,且与`slow`相遇(它们都指向同一个节点),那么链表中存在。 下面是伪代码形式的实现: ```java public boolean hasCycle(ListNode head) { ListNode slow = head; ListNode fast = head; // 如果链表为空或只有一个节点,不存在 if (head == null || head.next == null) { return false; } // 快慢指针开始遍历 while (fast != null && fast.next != null) { slow = slow.next; fast = fast.next.next; // 如果快指针先到末尾,则链表无 if (fast == null) { return false; } // 如果快慢指针相遇,说明链表有 if (slow == fast) { return true; } } return false; // 如果没有提前结束循,说明链表无 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值