IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> leetcode 497 528. Random Point in Non-overlapping Rectangles | 497. 非重叠矩形中的随机点(Java) -> 正文阅读

[数据结构与算法]leetcode 497 528. Random Point in Non-overlapping Rectangles | 497. 非重叠矩形中的随机点(Java)

497. Random Point in Non-overlapping Rectangles

https://leetcode.com/problems/random-point-in-non-overlapping-rectangles/
在这里插入图片描述

题解

大坑:面积包含边界,计算要注意+1,这种随机数的问题,没有固定的输出,调bug不好调啊…目前的方法是,手动指定random的值,遍历所有可能的random,看是否符合预期。

思路:
1、先按照面积计算每个rectangle的权重
2、根据权重选择rectangle的下标(参考 528. Random Pick with Weight
3、选好rectangle后,根据范围随机生成横坐标x、纵坐标y

class Solution {
    Random r;
    int N;
    int[][] rects;
    long[] weight; // 权重
    long[] weightSum; // 权重前缀和

    public Solution(int[][] rects) {
        this.rects = rects;
        r = new Random();
        N = rects.length;
        weight = new long[N];
        weightSum = new long[N];
        for (int i = 0; i < N; i++) {
            weight[i] = ((long) rects[i][2] + 1 - rects[i][0]) * ((long) rects[i][3] + 1 - rects[i][1]); // +1 因为包含边界!坑!!
        }
        weightSum[0] = 0;
        if (N > 1) weightSum[1] = weight[0];
        for (int i = 2; i < N; i++) {
            weightSum[i] = weightSum[i - 1] + weight[i - 1];
        }
    }

    public int[] pick() {
        // 根据权重选择一个rectangle
        long sum = weight[N - 1] + weightSum[N - 1];
        long rand = randLong(0, sum);
        // 反向查找randLong属于的index
        int L = 0;
        int R = N - 1;
        int index = -1;
        if (weightSum[N - 1] <= rand) {
            index = N - 1;
        } else {
            while (L < R) {
                int M = L + (R - L) / 2;
                if (weightSum[M] == rand) {
                    index = M;
                    break;
                } else if (weightSum[M] < rand) {
                    if (L == M) {
                        index = M;
                        break;
                    }
                    L = M;
                } else {
                    R = M - 1;
                    if (weightSum[R] <= rand) {
                        index = R;
                        break;
                    }
                }
            }
            if (index == -1) index = L;
        }
        int[] rec = rects[index];
        long x = randLong(rec[0], rec[2] + 1);
        long y = randLong(rec[1], rec[3] + 1);
        return new int[]{(int) x, (int) y};
    }

    public long randLong(long a, long b) { // 左闭右开
        return a + (long) (Math.random() * (b - a));
    }

}

/**
 * Your Solution object will be instantiated and called as such:
 * Solution obj = new Solution(rects);
 * int[] param_1 = obj.pick();
 */

在这里插入图片描述

528. Random Pick with Weight

https://leetcode.com/problems/random-pick-with-weight/
在这里插入图片描述
本题是上一题的一个子问题,直接复制粘贴了。

class Solution {
    int N;
    int[] w; // 权重
    int[] wSum; // 权重前缀和

    public Solution(int[] w) {
        N = w.length;
        this.w = w;
        wSum = new int[N];
        wSum[0] = 0;
        if (N > 1) wSum[1] = this.w[0];
        for (int i = 2; i < N; i++) {
            wSum[i] = wSum[i - 1] + this.w[i - 1];
        }
    }

    public int pickIndex() {
        // 根据权重选择一个rectangle
        int sum = w[N - 1] + wSum[N - 1];
        int rand = rand(0, sum);
        // 反向查找randLong属于的index
        int L = 0;
        int R = N - 1;
        int index = -1;
        if (wSum[N - 1] <= rand) {
            index = N - 1;
        } else {
            while (L < R) {
                int M = L + (R - L) / 2;
                if (wSum[M] == rand) {
                    index = M;
                    break;
                } else if (wSum[M] < rand) {
                    if (L == M) {
                        index = M;
                        break;
                    }
                    L = M;
                } else {
                    R = M - 1;
                    if (wSum[R] <= rand) {
                        index = R;
                        break;
                    }
                }
            }
            if (index == -1) index = L;
        }
        return index;
    }

    public int rand(int a, int b) { // 左闭右开
        return (int) (a + (Math.random() * (b - a)));
    }
}

/**
 * Your Solution object will be instantiated and called as such:
 * Solution obj = new Solution(w);
 * int param_1 = obj.pickIndex();
 */
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2021-09-13 09:31:11  更:2021-09-13 09:31:50 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/17 13:47:39-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码