LeetCode题目

解题分析

这类题目可能的解空间在一个范围内,不停二分遍历可能解。找到一个可行解后还要继续找,直到“最小值”的可行解。类比到二分搜索算法中就是满足条件的最左值。

以”410. 分割数组的最大值“问题为例,可能解空间在“最大的单个元素”和“所有元素和”之间。然后二分查找可行解空间,一个可行解是分组数等于或者小于目标组数。“小于目标组数”成立的原因是可以继续拆分,拆分后子数组元素和一定满足条件,显然这不一定是最优解。找到可行解后如果继续优化条件不成立,那么上一次的可行解就是最优解。

从可行解中找最左值“二分搜索”中边界条件很容易出错。比如下面找最左值的方式:

        while (left < right) {
            int mid = left + (right - left) / 2;
 
            int splits = split(nums, mid);
            if (splits > m) {
                // 如果分割数太多,说明「子数组各自的和的最大值」太小,此时需要将「子数组各自的和的最大值」调大
                // 下一轮搜索的区间是 [mid + 1, right]
                left = mid + 1;
            } else {
                // 下一轮搜索的区间是上一轮的反面区间 [left, mid]
                right = mid;
            }
        }

个人更喜欢下面的方式,容易理解,不易出错,缺点是代码有点冗余。

while(left <= right) {
	int mid = left + (right - left) / 2;
	int groups = split(nums, mid);
 
	if(groups > k) { // 分组和太小,分组数大于预定值
		left = mid + 1;
	} else if(groups < k) { // 分组和太大,分组数小于预定值
		right = mid - 1;
	} else { // 可能分组数 < k 走不到这里,因为 split 是贪心算法。但实际上可以从 < k 拆分到 k 组
		if(mid == max || split(nums, mid -1) > k) {
			return mid;
		} else {
			right = mid - 1;
		}
	}
}
package org.stone.study.algo.ex202403;  
  
/**  
 * [410. 分割数组的最大值](https://leetcode.cn/problems/split-array-largest-sum/)
 * * 给定一个非负整数数组 nums 和一个整数 k ,你需要将这个数组分成 k 个非空的连续子数组。  
 * 设计一个算法使得这 k 个子数组各自和的最大值最小。  
 */  
public class SplitArrayWithMinSum {  
  
    public static void main(String[] args) {  
        int[] nums = new int[] {7,2,5,10,8};  
        int k = 2;  
        // ans: 18  
        System.out.println("ans:" + new SplitArrayWithMinSum().splitArray(nums, k));  
    }  
  
    /**  
     * 给定一个非负整数数组 nums 和一个整数 k ,把这个数组分成 k 个非空的连续子数组。  
     * 找到一种分组方式,使得返回k 个子数组各自和的最大值最小  
     * @param nums  
     * @param k  
     * @return  
     */  
    public int splitArray(int[] nums, int k) {  
        int max = 0, sum = 0;  
        for(int num : nums) {  
            max = Math.max(max, num);  
            sum += num;  
        }  
  
        int left = max, right = sum;  
        while(left <= right) {  
            int mid = left + (right - left) / 2;  
  
            int groups = split(nums, mid);  
            //System.out.println("left:" + left + ", right:" + right + ", mid:" + mid + ", groups:" + groups);  
            if(groups > k) { // 分组和太小,分组数大于预定值  
                left = mid + 1;  
            } else if(groups < k) { // 分组和太大,分组数小于预定值  
                right = mid - 1;  
            } else { // 可能分组数 < k 走不到这里,因为 split 是贪心算法。但实际上可以从 < k 拆分到 k 组  
                if(mid == max || split(nums, mid -1) != k) {  
                    return mid;  
                } else {  
                    right = mid - 1;  
                }  
            }  
        }  
  
        return left;  
    }  
  
    /**  
     * 拆分数组,每组和最大为maxSum。返回可以拆分的组数  
     * 直接贪心算法实现  
     */  
    private int split(int[] nums, int maxSum) {  
        int curSum = 0;  
        int splitNum = 1;  
  
        for(int num : nums) {  
            if(curSum + num > maxSum) {  
                ++splitNum;  
                curSum = 0;  
            }  
  
            curSum += num;  
        }  
  
        return splitNum;  
    }  
}
package org.stone.study.algo.ex202403;  
  
import java.util.Arrays;  
  
/**  
 * [875. 爱吃香蕉的珂珂](https://leetcode.cn/problems/koko-eating-bananas/)
 * 珂珂喜欢吃香蕉。这里有 n 堆香蕉,第 i 堆中有 piles[i] 根香蕉。警卫已经离开了,将在 h 小时后回来。  
 * 珂珂可以决定她吃香蕉的速度 k (单位:根/小时)。每个小时,她将会选择一堆香蕉,从中吃掉 k 根。如果这堆香蕉少于 k 根,她将吃掉这堆的所有香蕉,然后这一小时内不会再吃更多的香蕉。  
 * 珂珂喜欢慢慢吃,但仍然想在警卫回来前吃掉所有的香蕉。  
 * 返回她可以在 h 小时内吃掉所有香蕉的最小速度 k(k 为整数)。  
 */  
public class MinEatingSpeed {  
    /**  
     * 解法 1:二分查找最左值边界处理好理解一点。  
     * @param piles  
     * @param h  
     * @return  
     */  
    public int minEatingSpeed(int[] piles, int h) {  
        int max = Arrays.stream(piles).max().getAsInt();  
  
        int left = 1, right = max;  
        while(left <= right) {  
            int mid = left + (right - left) / 2;  
  
            long spent = spentHours(piles, mid);  
  
            if(spent > h) {  
                left = mid + 1;  
            } else if(spent < h){  
                right = mid - 1;  
            } else {  
                //System.out.println("spent:" + spent + ", left:" + left + ", right:" + right);  
                if(mid == 1 || spentHours(piles, mid - 1) > h) {  
                    return mid;  
                } else {  
                    right = mid - 1;  
                }  
            }  
        }  
        //System.out.println("left:" + left + ", right:" + right);  
        return left;  
    }  
  
    /**  
     * 以速度 speed 依次消耗 piles 数组。不足去整(往大的取)  
     */  
    private long spentHours(int[] piles, int speed) {  
        long spent = 0;  
        for(int pile : piles) {  
            spent += (pile + speed - 1) / speed;  
        }  
  
        return spent;  
    }  
  
  
    /**  
     * 解法 2,更简洁,但是二分查找最左值边界理解上有点困难  
     * @param piles  
     * @param h  
     * @return  
     */  
    public int minEatingSpeed2(int[] piles, int h) {  
        int max = 0;  
        for(int p : piles) {  
            max = Math.max(max, p);  
        }  
  
        int l = 1, r = max, ans = max;  
        while(l <= r) {  
            int total = 0;  
            int m = l + (r -l) / 2;  
            for(int p : piles) {  
                total += (p-1)/m + 1; //向上取整  
            }  
  
            if(total > h) {  
                l = m + 1;  
            } else {  
                ans = m;  
                r = m - 1;  
            }  
        }  
  
        return ans;  
    }  
}

通用的二分查找值,最左值,最右值方法请参考1

 
package org.stone.study.algo.search;  
  
/**  
 * @Author: shidonghua  
 * @Description:  
 * @Date: 3/8/21 09:07  
 * @Version: 1.0  
 */public class BinarySearch {  
  
    /**  
     * 二分搜索,并找最左边的元素  
     *  
     * @param a  
     * @param n  
     * @param value  
     * @return  
     */  
    public int bsearch_LeftMost(int[] a, int n, int value) {  
        int low = 0;  
        int high = n - 1;  
        while (low <= high) {  
            int mid = low + ((high - low) >> 1);  
            if (a[mid] > value) {  
                high = mid - 1;  
            } else if (a[mid] < value) {  
                low = mid + 1;  
            } else {  
                if ((mid == 0) || (a[mid - 1] != value)) return mid; // 最左逼近  
                else high = mid - 1;  
            }  
        }  
        return -1;  
    }  
  
    /**  
     * 二分搜索最右边的元素  
     *  
     * @param a  
     * @param n  
     * @param value  
     * @return  
     */  
    public int bsearch_RightMost(int[] a, int n, int value) {  
        int low = 0;  
        int high = n - 1;  
        while (low <= high) {  
            int mid = low + ((high - low) >> 1);  
            if (a[mid] > value) {  
                high = mid - 1;  
            } else if (a[mid] < value) {  
                low = mid + 1;  
            } else {  
                if ((mid == n - 1) || (a[mid + 1] != value)) return mid; // 往右边逼近  
                else low = mid + 1;  
            }  
        }  
        return -1;  
    }  
  
    /**  
     * 第一个大于等于元素  
     *  
     * @param a  
     * @param n  
     * @param value  
     * @return  
     */  
    public int bsearch_GE(int[] a, int n, int value) {  
        int low = 0;  
        int high = n - 1;  
        while (low <= high) {  
            int mid = low + ((high - low) >> 1);  
            if (a[mid] >= value) {  
                if ((mid == 0) || (a[mid - 1] < value)) return mid;  
                else high = mid - 1;  
            } else {  
                low = mid + 1;  
            }  
        }  
        return -1;  
    }  
  
    /**  
     * 最后一个小于等于  
     *  
     * @param a  
     * @param n  
     * @param value  
     * @return  
     */  
    public int bsearch_LE(int[] a, int n, int value) {  
        int low = 0;  
        int high = n - 1;  
        while (low <= high) {  
            int mid = low + ((high - low) >> 1);  
            if (a[mid] > value) {  
                high = mid - 1;  
            } else {  
                if ((mid == n - 1) || (a[mid + 1] > value)) return mid;  
                else low = mid + 1;  
            }  
        }  
        return -1;  
    }  
}

总结

二分搜索需要数组满足单调性,简单等值搜索的场景比较少,实际场景更多是最值搜索,映射到算法中就是满足条件的最左或者最右值。这时处理算法的边界条件很重要,也是最容易出错的地方。

算法题一般不会直接让写二分搜索,需要把具体问题映射到“二分搜索”的解题思路上。比如解空间在一个单调整数数组上,找最小解等。

Footnotes

  1. github