python - 为什么这个随机生成的球形点云分布不均匀?
问题描述
我正在尝试模拟从点源发射的辐射。为此,给定光源的坐标和所需的发射光线长度,我在球坐标中随机生成一个方向向量,将其转换为笛卡尔坐标,并返回正确的终点。但是,当我运行它并在 Blender 中可视化生成的点云(由所有随机生成的端点组成)时,我发现它在球体的“极点”处的人口密度更高。我希望点沿球体均匀分布。我怎样才能做到这一点?
随机生成函数:
def getRadiationEmissionLineSeg(p, t):
if(p.size == 4):
#polar angle spans [0, pi] from +Z axis to -Z axis
#azimuthal angle spans [0, 2*pi] orthogonal to the zenith (in the XY plane)
theta = math.pi * random.random()
phi = 2 * math.pi * random.random()
#use r = 1 to get a unit direction vector
v = sphericalToCartesian(1, theta, phi)
#parametric vector form: vec = p + tv
#p = point that lies on vector (origin point in case of a ray)
#t = parameter (-inf, inf) for lines, [0, inf) for rays
#v = direction vector (must be normalized)
return p + t * v
球坐标->笛卡尔转换函数:
def sphericalToCartesian(r, theta, phi):
x = r * math.sin(theta) * math.cos(phi)
y = r * math.sin(theta) * math.sin(phi)
z = r * math.cos(theta)
return npy.array([x, y, z, 0])
解决方案
当您通过球坐标变换点并且角度 theta 接近 pi 时,作为 [0,2pi]x{theta} 的图像的圆变得越来越小。由于 theta 是均匀分布的,所以极点附近的点会更多。它可以在网格图像上看到。
如果要在球体上生成均匀分布的点,可以使用以下事实:如果用两个平行平面切割球体,则平面之间的球面带的面积仅取决于平面之间的距离。因此,您可以使用两个均匀分布的随机变量在球体上获得均匀分布:
- -r 和 r 之间的 z 坐标,
- [0, 2pi) 之间的角度 theta 对应于经度。
然后您可以轻松计算 x 和 y 坐标。
示例代码:
import numpy as np
import matplotlib.pyplot as plt
from mpl_toolkits.mplot3d import Axes3D
r = 1
n = 1000
z = np.random.random(n)*2*r - r
phi = np.random.random(n)*2*np.pi
x = np.sqrt(1 - z**2)*np.cos(phi)
y = np.sqrt(1 - z**2)*np.sin(phi)
fig = plt.figure(figsize=(8, 8))
ax = plt.axes(projection='3d')
ax.scatter(x, y, z)
plt.show()
结果n=100,250,1000
:
推荐阅读
- javascript - JavaScript .closest 返回 null
- java - 如何杀死在服务器启动时声明的线程启动无限循环
- caching - Azure CDN“丢失”请求
- javascript - 两组之间的 Phaser 3 对撞机不触发
- pytorch - 未设置 --gpus 时防止在 SLURM 中使用 GPU
- terminal - 如何将我的文本文件输出为大写哈希?
- javascript - 在 React JS 函数中更新数组中每个对象的属性
- java - 由公共类 MyApplication 扩展 Application 引起的 java.lang.NullPointerException
- compiler-errors - Ada - 一维数组操作
- javascript - 重新渲染我的反应组件时我的动画不起作用?