478. Generate Random Point in a Circle
1. 问题
给定一个圆的半径和圆心坐标,生成圆内点的坐标。
2. 思路
简单说
(1)在圆内随机取点不好做,但是如果画出这个圆的外接正方形,在正方形里面采样就好做了。
(2)取两个random确定正方形内的横坐标和纵坐标即可在正方形内采样。
(3)如果采样到的点不在圆内,则丢弃,继续采样,当采样的点在圆内,则返回该点。
(4)这样采样可以视作在圆内采样的一种近似,到这里就可以把问题解决了。下面是一些扩展知识。
拒绝采样(Reject Sampling)的解释
这个方法利用了拒绝采样的方法,该方法的详细阐述我在前一篇文章470. Implement Rand10() Using Rand7() (拒绝采样Reject Sampling)中提过,下面的解释以这里面拒绝采样的介绍为基础,沿用里面的符号和式子。
把圆内点的分布当作p(x),圆内可以看作有 \(\pi r^2\)个点,每个点的概率为 $ 1 / (\pi r^2) $。
然后把正方形内点的分布当作q(x),正方形内可以看作有 \(4r^2\)个点,每个点的概率为 $1 / (4r^2) $。
这里我们M取\(4/\pi\)。
我们在\(q(x)\)上采点,每个点采到的概率为$1 / (4r^2) $,乘上M就是 \(1 / (\pi r^2)\),对于每个随机变量,\(M * q(x)\)为\(1 / (\pi r^2)\)。\(p(x)\)是圆内点的分布,当采到的点在圆内时,\(p(x) = 1 / (\pi r^2)\),那么接受概率为\(p(x) / (M * q(x))\)就是1。当采到的点在圆外时,\(p(x)\)为0,所以接受概率为0。
时间复杂度:平均为O(1),最坏情况O(无穷)。
空间复杂度:O(1)。
random调用次数的期望值:2.55
这个期望值的计算参见470. Implement Rand10() Using Rand7() (拒绝采样Reject Sampling),使用错位相减,等比数列的求和以及极限,最后求出是\(2 * \frac{4}{\pi} = 2.55\)。
(方法二)直接在圆上取点
(1)一个半径为R的圆,可以看成无数个半径为r(r<R)的圆组成,我们只需要随机取一个半径,然后随机取一个角度,利用半径和角度即可随机取到我们要的点。
(2)对于角度,随机从\([0, 2\pi]\)取就可以了。点的横坐标和纵坐标需要根据这个角度的sin和cos来得到。
(3)对于半径而言,很容易误以为从\([0,1]\)取个数然后乘以R就可以了,但是我们知道,从半径为r的圆内取点,和从半径为\(R\)的圆内取点,比例是\((r/R)^2\)而不是\(r/R\)。比如说我们在半径为\(R\)的圆内取到半径为\((1/2)R\)的圆内点的概率应该\(1/4\)而不是\(1/2\)。所以应该\([0,1]\)取个数,开个根号之后再乘以R,这样保证了在圆内的等比例采样。
时间复杂度:O(1)
空间复杂度:O(1)
3. 代码
拒绝采样
class Solution(object):
def __init__(self, radius, x_center, y_center):
"""
:type radius: float
:type x_center: float
:type y_center: float
"""
self.radius = radius
self.x_center = x_center
self.y_center = y_center
def randPoint(self):
"""
:rtype: List[float]
"""
while True:
x = (self.x_center - self.radius) + random.random() * self.radius * 2
y = (self.y_center - self.radius) + random.random() * self.radius * 2
if (x - self.x_center) ** 2 + (y - self.y_center) ** 2 <= self.radius ** 2:
return [x, y]
直接在圆上取点
import random
import math
class Solution(object):
def __init__(self, radius, x_center, y_center):
self.radius = radius
self.x_center = x_center
self.y_center = y_center
def randPoint(self):
r = math.sqrt(random.random()) * self.radius
theta = random.uniform(0, 2*math.pi)
x = self.x_center + r*math.cos(theta)
y = self.y_center + r*math.sin(theta)
return [x, y]
4. 类似题目
470. Implement Rand10() Using Rand7() (拒绝采样Reject Sampling)
519. Random Flip Matrix
478. Generate Random Point in a Circle的更多相关文章
- 【LeetCode】478. Generate Random Point in a Circle 解题报告(Python & C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题目地址: https://leetcode.com/problems/generate ...
- [LeetCode] Generate Random Point in a Circle 生成圆中的随机点
Given the radius and x-y positions of the center of a circle, write a function randPoint which gener ...
- [Swift]LeetCode478. 在圆内随机生成点 | Generate Random Point in a Circle
Given the radius and x-y positions of the center of a circle, write a function randPoint which gener ...
- Java – Generate random integers in a rangejava获取某个范围内的一个随机数
In this article, we will show you three ways to generate random integers in a range. java.util.Rando ...
- #227 Generate Random Whole Numbers within a Range
我们之前生成的随机数是在0到某个数之间,现在我们要生成的随机数是在两个指定的数之间. 我们需要定义一个最小值和一个最大值. 下面是我们将要使用的方法,仔细看看并尝试理解这行代码到底在干嘛: Math. ...
- Gym 100956 A Random Points on the Circle
二分答案. 对于每次二分后的答案来说, 先倍增序列,通过 two point 来找到 以每个点为起点的最优的符合答案的在哪里. 然后可以DFS树去判断他的前k祖先之间的距离是不是大于k. 常数有点大. ...
- 519. Random Flip Matrix(Fisher-Yates洗牌算法)
1. 问题 给定一个全零矩阵的行和列,实现flip函数随机把一个0变成1并返回索引,实现rest函数将所有数归零. 2. 思路 拒绝采样 (1)先计算矩阵的元素个数(行乘以列),记作n,那么[0, n ...
- 470. Implement Rand10() Using Rand7() (拒绝采样Reject Sampling)
1. 问题 已提供一个Rand7()的API可以随机生成1到7的数字,使用Rand7实现Rand10,Rand10可以随机生成1到10的数字. 2. 思路 简单说: (1)通过(Rand N - 1) ...
- 【LeetCode】随机化算法 random(共6题)
[384]Shuffle an Array(2019年3月12日) Shuffle a set of numbers without duplicates. 实现一个类,里面有两个 api,struc ...
随机推荐
- RIDE指定log和report的输出目录
在命令行中,输入 pybot --help就可以看到他支持的所以命令和相关的介绍 我们可以看到outputdir这个命令,就是来知道report和log的输出目录的 如果你是在命令行中,那么直接后面跟 ...
- Activity、Window和View三者间的关系有一定的见解
一.简述如何将Activity展现在手机上 Tips: Activity本身是没办法处理显示什么控件(view)的,是通过PhoneWindow进行显示的 换句话说:activity就是在造Phone ...
- C++说明符和限定符
有些被称为存储说明符(storage class specifier)或cv-限定符(cv-qualifier)的C++关键字提供了一些有关存储的信息.下面是存储所说明符:* auto (在C++11 ...
- Java自动类型转换
■ 自动类型转换:容量小的数据类型可以自动转换为容量大的数据类型. ■ 特例:可以讲整型常量直接赋给byte,short,char等类型变量,而不需要强制类型转换,只要不超出其表数范围. ■ 强制类型 ...
- img-图片二进制流 64位前端显示
碰到的场景:因为使用iframe子窗口打开,多张的二维码图片创建方法调用,导致页面打开缓慢, 所以将调取方式转换成<img src="data:image/png;base64,@it ...
- 学习POC框架pocsuite--编写hellowordPOC
在这里,首先向安全圈最大的娱乐公司,某404致敬. 参考博文 https://www.seebug.org/help/dev 向seebug平台及该文原作者致敬,虽然并不知道是谁 长话短说其实,可自由 ...
- 【BZOJ1458】士兵占领 最小流
[BZOJ1458]士兵占领 Description 有一个M * N的棋盘,有的格子是障碍.现在你要选择一些格子来放置一些士兵,一个格子里最多可以放置一个士兵,障碍格里不能放置士兵.我们称这些士兵占 ...
- 微信小程序 --- 登录页面
思路:在个人中心页面,首先判断全局的 app.js 里面的 globalData 里面的 is_login 状态,并且判断 缓存数据中的 is_login 状态,如果都为真,就正常显示,如果有一个为假 ...
- 获取鼠标经过位置的X、Y坐标
利用JavaScript获取鼠标经过位置的X.Y坐标方法. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN& ...
- 6.IIs部署与发布
A.网站的发布步骤: 1.首先要选择要发布的网站(即项目里的网站)也就是代码. 2.左键选择发布. 3.配置文件:Web.congig. 4.连接:publis method:File System, ...