[leetcode/lintcode 题解] 百度面试真题:木材加工

简介: [leetcode/lintcode 题解] 百度面试真题:木材加工

描述
有一些原木,现在想把这些木头切割成一些长度相同的小段木头,需要得到的小段的数目至少为k。当然,我们希望得到的小段越长越好,你需要计算能够得到的小段木头的最大长度。
木头长度的单位是厘米。原木的长度都是正整数,我们要求切割得到的小段木头的长度也要求是整数。无法切出要求至少k段的,则返回0即可。

在线评测地址:领扣题库官网

样例1
输入:
L = [232, 124, 456]
k = 7
输出: 114
Explanation: 我们可以把它分成114cm的7段,而115cm不可以
样例2
输入:
L = [1, 2, 3]
k = 7
输出: 0
说明:很显然我们不能按照题目要求完成。

算法:二分

题目意思是说给出n段木材L[i], 将这n段木材切分为至少k段,这k段等长,
若直接枚举每段木材的长度则时间复杂度高达 O(n*maxL), 我们可以使用二分答案来优化枚举木材长度的过程

  • 设left=0,即木材长度最小为0,设right=max_L即所有木材中最长的长度,因为结果是不可能大于这个长度的,mid = left + right
  • 若长度为mid时不能完成,说明太长了,那么我们往区间[left,mid]搜,
  • 若可以完成,说明也许可以更长,那么我们往[mid,right]搜,
  • 在check函数中,我们判断用所有木头除当前mid的值的和是否大于等于k,若小于则说明该mid不可行, 若大于等于则说明mid可行
  • 由于判断条件是left + 1 < right,最后结果就是left的值

复杂度分析

  • 时间复杂度O(nlog(n))

    • 二分查找的复杂度
  • 空间复杂度O(size(L))

    • 只有数组L
public class Solution {
    /**
     * @param L: Given n pieces of wood with length L[i]
     * @param k: An integer
     * @return: The maximum length of the small pieces
     */
    public int woodCut(int[] L, int k) {
        int len_L = L.length;
        if (len_L == 0) {
            return 0;
        }
        int max_L = 0;//寻找最大值
        for (int i = 0;i < len_L;i++) {
            max_L = Math.max(max_L,L[i]);
        }
        long left = 0,right = max_L;
        int mid;
        while (left + 1 < right) {
            mid = (int)(left + (right - left) / 2);
            if (check(L, k, len_L, mid)) {//如果还能分更长的,则往[mid,right]走
                left = mid;
            }
            else {//如果不能分更长的,则往[left,mid]走
                right = mid;
            }
        }
        if(check(L,k,len_L,(int)right)) return (int)right;
        return (int)left;
    }
    private boolean check(int [] L,int k,int len_L,int mid){
        int count = 0;
        //计算当前长度下能分成几段
        for (int i = 0;i < len_L;i++) {
            count += L[i] / mid;
        }
        //如果还能分更长的,返回true
        if (count >= k) {
            return true;
        }
        //如果不能分更长的,返回false
        return false;
    }
}

更多题解参考:九章官网solution

相关文章
|
1月前
|
算法
【数组相关面试题】LeetCode试题
【数组相关面试题】LeetCode试题
|
1月前
力扣面试经典题之哈希表
力扣面试经典题之哈希表
18 0
|
1月前
|
存储
力扣面试经典题之数组/字符串
力扣面试经典题之数组/字符串
23 0
|
3月前
|
存储 算法 程序员
【Leetcode 程序员面试金典 01.01】判定字符是否唯一 —— 位运算|哈希表
可以使用哈希表或位运算来解决此问题:由题可知s[i]仅包含小写字母,int[26]即能表示字符的出现次数;
|
3月前
|
算法 程序员 索引
【Leetcode 程序员面试金典 02.08】 —— 环路检测 |双指针
我们可以使用双指针解决本题,由数学推导可知:a 的距离为(环长度的倍数 - b),即 tmp 指针从头节点走到环开头节点等于 slow 指针走到环开头节点的距离
|
3月前
|
Java 程序员
【Leetcode 程序员面试金典 05.01】插入 —— 位运算
位运算问题,只需要把 N 的 i 到 j 位都置 0 后再和 M 左移 i 位的结果进行按位或即可
|
1月前
力扣面试经典题之二叉树
力扣面试经典题之二叉树
16 0
|
2月前
|
JavaScript 前端开发 C语言
leetcode每日一题 2021/4/2 面试题 17.21. 直方图的水量
leetcode每日一题 2021/4/2 面试题 17.21. 直方图的水量
23 0
|
23天前
|
算法
【力扣经典面试题】121. 买卖股票的最佳时机
【力扣经典面试题】121. 买卖股票的最佳时机
|
23天前
|
存储
【力扣经典面试题】80. 删除有序数组中的重复项 II
【力扣经典面试题】80. 删除有序数组中的重复项 II