Leetcode 528 按权重随机选择

题目信息

LeetoCode地址: . - 力扣(LeetCode)

题目理解

想象题目提供的w数组里是很多根长短不一的棍子,然后我们将其按顺序排列成一条线。

然后我们扔一个沙包,砸中哪一根棍子,就代表命中了那根棍子代表的数字。很显然,棍子越长,就越容易砸中。

假如这五根棍子分别长1,2,3,4,5,那么合并后总长度就是1+2+3+4+5=15

那么沙包扔出后可能会落在0到15之间的任何一个位置。具体来说,落到[0,1]的可能性会比落到[1,3]小,因为后者代表的区间更大,而概率最大的则是[10,15],这代表了最后一根棍子的区间。

那么题目就变成了,如何快速找到沙包落点属于哪一个区间。使用前缀和,我们可以很轻松的表示这5根棍子的区间:[1,3,6,10,15],而这又恰巧是一个单调递增数组,使用二分法,可以在O(logn)的时间复杂度内找到区间。

前缀和+二分写法

java 复制代码
class Solution {
    int[] preSum;
    Random random;
    int l;
    public Solution(int[] w) {
        l = w.length;
        random = new Random();
        preSum = new int[l];
        preSum[0] = w[0];
        for (int i = 1; i<l; i++) {
            preSum[i] = preSum[i-1] + w[i];
        }
    }
    
    public int pickIndex() {
        int num = random.nextInt(preSum[l-1])+1;
        int left = 0, right = l-1;
        while (left < right) {
            int mid = left + (right-left)/2;
            if (preSum[mid] == num) {
                return mid;
            }
            if (preSum[mid] < num) {
                left = mid+1;
            } else {
                right = mid-1;
            }
        }
        return preSum[left] >=  num ? left : left+1;
    }
}

/**
 * Your Solution object will be instantiated and called as such:
 * Solution obj = new Solution(w);
 * int param_1 = obj.pickIndex();
 */

写法2

相关推荐
专业抄代码选手39 分钟前
【Leetcode】1930. 长度为 3 的不同回文子序列
javascript·算法·面试
[J] 一坚1 小时前
深入浅出理解冒泡、插入排序和归并、快速排序递归调用过程
c语言·数据结构·算法·排序算法
czlczl200209251 小时前
算法:二叉搜索树的最近公共祖先
算法
司铭鸿1 小时前
祖先关系的数学重构:从家谱到算法的思维跃迁
开发语言·数据结构·人工智能·算法·重构·c#·哈希算法
程序员小远2 小时前
Appium-移动端自动测试框架详解
自动化测试·软件测试·python·测试工具·职场和发展·appium·测试用例
SoleMotive.2 小时前
redis实现漏桶算法--https://blog.csdn.net/m0_74908430/article/details/155076710
redis·算法·junit
-森屿安年-2 小时前
LeetCode 283. 移动零
开发语言·c++·算法·leetcode
北京地铁1号线3 小时前
数据结构:堆
java·数据结构·算法
散峰而望3 小时前
C++数组(一)(算法竞赛)
c语言·开发语言·c++·算法·github
自然常数e3 小时前
深入理解指针(1)
c语言·算法·visual studio