二分算法到红蓝染色

我的学习来源主要是灵神的力扣题解

下面是一个简单的二分查找的代码

复制代码
java 复制代码
public static int findPeakElement(int[] nums) {
        int left = 0;
        int right = nums.length - 2; // 初始右边界为倒数第二个元素
        
        while (left <= right) {
            int mid = left + (right - left) / 2; // 防止整数溢出的中间值计算
            
            // 比较中间元素和右侧元素
            if (nums[mid] < nums[mid + 1]) {
                left = mid + 1; // 峰值在右侧,调整左边界
            } else {
                right = mid - 1; // 峰值在左侧或当前元素为峰,调整右边界
            }
        }
        
        return left; // 最终left指向峰值位置
    }

这个代码很常见,每次去取mid然后比较就ok了,适用于有序数组。时间复杂度logn

之前做过一家大厂笔试题,其中有道题叫找出可能存在的二分序列有下面几个

10 20 30 40

40 10 20 30

20 30 40 10

40 20 10 30

正确的答案应该是第一个和第二个,二分序列就是每次的mid值

为什么呢

因为这个二分查找有一个隐藏性质和快速排序类似,就是每一次操作之后,左边小于mid,右边大于mid。

等于的情况得自己定义了

这个性质看似很简单,但是却延申出了用二分做算法题的一个常见方法

红蓝染色法

红蓝染色法

left指针掌管左边蓝色区域,蓝色表示 truej即所求值及其右侧,right指针掌管右边红色区域,红色表示false即所求值左侧,两者互不冲突,通过不断向目标区域靠近,扩大两个指针的掌管区域,直到两者掌管的区域接壤

使用闭区间时,L-1必定是红色即false,R+1必定是蓝色即true,这就是循环不变量

关键不在于区间里的元素具有什么性质,而是区间外面的元素具有什么性质,即将区间外染成红色与蓝色。

根据以上两点,在做题时关键即确定二分条件函数isBlue(),判断是否满足条件,满足条件则right 左移使右侧变蓝 ,不满足条件则left 右移使左侧变红

核心思想就是去查找一个数时,他的左右必然是不满足同一个法则的,有规律1的染红,有规律2的染蓝,交界处即为所求。

LeetCode162. 寻找峰值这道题

java 复制代码
public static int findPeakElement(int[] nums) {
        int left = 0;
        int right = nums.length - 2; // 初始右边界为倒数第二个元素
        
        while (left <= right) {
            int mid = left + (right - left) / 2; // 防止整数溢出的中间值计算
            
            // 比较中间元素和右侧元素
            if (nums[mid] < nums[mid + 1]) {
                left = mid + 1; // 峰值在右侧,调整左边界
            } else {
                right = mid - 1; // 峰值在左侧或当前元素为峰,调整右边界
            }
        }
        
        return left; // 最终left指向峰值位置
    }

这个时候红色的规则是什么

nums[mid] < nums[mid + 1]

蓝色的规则呢

nums[mid] > nums[mid + 1]

这道题还比较简单,下面再看一道,懂了应该就能明白这个办法了

LeetCode33. 搜索旋转排序数组

先找到旋转数组的最小值

即min的位置,然后比较目标值与E的大小

再使用正常的二分搜索

复制代码
java 复制代码
public class RotatedBinarySearch {

 
    public static int findMin(int[] nums) {
        int left = 0;
        int right = nums.length - 2; // 初始右边界为倒数第二个元素
        
        while (left <= right) {
            int mid = left + (right - left) / 2; // 防止溢出的中间值计算
            
            // 比较中间元素和最后一个元素
            if (nums[mid] < nums[nums.length - 1]) {
                right = mid - 1; // 最小值在左半区
            } else {
                left = mid + 1; // 最小值在右半区
            }
        }
        
        return left; // 返回最小值索引
    }

   
    public static int lowerBound(int[] nums, int left, int right, int target) {
        int r0 = right; // 保存初始右边界
        
        while (left <= right) {
            int mid = left + (right - left) / 2;
            
            if (nums[mid] < target) {
                left = mid + 1; // 目标在右半区
            } else {
                right = mid - 1; // 目标在左半区
            }
        }
        
        // 检查是否越界或未找到
        if (left == r0 + 1 || nums[left] != target) {
            return -1;
        }
        return left;
    }

    public static int search(int[] nums, int target) {
        int i = findMin(nums);
        
        // 根据目标值与末尾元素比较决定搜索区间
        if (target > nums[nums.length - 1]) {
            return lowerBound(nums, 0, i - 1, target); // 搜索左半区
        }
        return lowerBound(nums, i, nums.length - 1, target); // 搜索右半区
    }

这里主要说说找最小值的染色

红色:nums[mid] > nums[nums.length - 1]

蓝色:nums[mid] < nums[nums.length - 1]

相关推荐
不知天地为何吴女士1 小时前
Day32| 509. 斐波那契数、70. 爬楼梯、746. 使用最小花费爬楼梯
算法
小坏坏的大世界1 小时前
C++ STL常用容器总结(vector, deque, list, map, set)
c++·算法
我命由我123452 小时前
Kotlin 数据容器 - List(List 概述、创建 List、List 核心特性、List 元素访问、List 遍历)
java·开发语言·jvm·windows·java-ee·kotlin·list
励志要当大牛的小白菜4 小时前
ART配对软件使用
开发语言·c++·qt·算法
qq_513970444 小时前
力扣 hot100 Day56
算法·leetcode
武子康4 小时前
Java-80 深入浅出 RPC Dubbo 动态服务降级:从雪崩防护到配置中心秒级生效
java·分布式·后端·spring·微服务·rpc·dubbo
PAK向日葵5 小时前
【算法导论】如何攻克一道Hard难度的LeetCode题?以「寻找两个正序数组的中位数」为例
c++·算法·面试
爱装代码的小瓶子6 小时前
数据结构之队列(C语言)
c语言·开发语言·数据结构
爱喝矿泉水的猛男7 小时前
非定长滑动窗口(持续更新)
算法·leetcode·职场和发展
YuTaoShao7 小时前
【LeetCode 热题 100】131. 分割回文串——回溯
java·算法·leetcode·深度优先