我是编程新手,所以我希望我的愚蠢问题不会打扰你。我现在正在尝试使用 python 计算泊松球分布(泊松盘的 3D 版本),然后将结果插入 POV-RAY 以便我可以生成一些随机分布的填充岩石。我正在关注这两个链接:[ https://github.com/CodingTrain/Rainbow-Code/blob/master/CodingChallenges/CC_33_poisson_disc/sketch.js#L13]
[ https://www.cs.ubc.ca/~ rbridson/docs/bridson-siggraph07-poissondisk.pdf]
tl;dr 0.创建一个 n 维网格数组,单元格大小 = r/sqrt(n) 其中 r 是每个球体之间的最小距离。所有数组都设置为默认 -1 代表“无点”
1.创建一个初始样本。(它应该是随机放置的,但我选择放在中间)。把它放在网格数组中。此外,初始化一个活动数组。将初始样本放入活动数组中。2.当活动列表不为空时,选择一个随机索引。在它附近生成点并确保这些点不与附近的点重叠(仅使用附近的数组进行测试)。如果在“随机索引”附近无法创建样本,则将“随机索引”踢出。循环这个过程。
这是我的代码:
import math
from random import uniform
import numpy
import random
radius = 1 #you can change the size of each sphere
mindis = 2 * radius
maxx = 10 #you can change the size of the container
maxy = 10
maxz = 10
k = 30
cellsize = mindis / math.sqrt(3)
nrofx = math.floor(maxx / cellsize)
nrofy = math.floor(maxy / cellsize)
nrofz = math.floor(maxz / cellsize)
grid = []
active = []
default = numpy.array((-1, -1, -1))
for fillindex in range(nrofx * nrofy * nrofz):
grid.append(default)
x = uniform(0, maxx)
y = uniform(0, maxy)
z = uniform(0, maxz)
firstpos = numpy.array((x, y, z))
firsti = maxx // 2
firstj = maxy // 2
firstk = maxz // 2
grid[firsti + nrofx * (firstj + nrofy * firstk)] = firstpos
active.append(firstpos)
while (len(active) > 0) :
randindex = math.floor(uniform(0,len(active)))
pos = active[randindex]
found = False
for attempt in range(k):
offsetx = uniform(mindis, 2 * mindis)
offsety = uniform(mindis, 2 * mindis)
offsetz = uniform(mindis, 2 * mindis)
samplex = offsetx * random.choice([1,-1])
sampley = offsety * random.choice([1,-1])
samplez = offsetz * random.choice([1,-1])
sample = numpy.array((samplex, sampley, samplez))
sample = numpy.add(sample, pos)
xcoor = math.floor(sample.item(0) / cellsize)
ycoor = math.floor(sample.item(1) / cellsize)
zcoor = math.floor(sample.item(2) / cellsize)
attemptindex = xcoor + nrofx * (ycoor + nrofy * zcoor)
if attemptindex >= 0 and attemptindex < nrofx * nrofy * nrofz and numpy.all([sample, default]) == True and xcoor > 0 and ycoor > 0 and zcoor > 0 :
test = True
for testx in range(-1,2):
for testy in range(-1, 2):
for testz in range(-1, 2):
testindex = (xcoor + testx) + nrofx * ((ycoor + testy) + nrofy * (zcoor + testz))
if testindex >=0 and testindex < nrofx * nrofy * nrofz :
neighbour = grid[testindex]
if numpy.all([neighbour, sample]) == False:
if numpy.all([neighbour, default]) == False:
distance = numpy.linalg.norm(sample - neighbour)
if distance > mindis:
test = False
if test == True and len(active)<len(grid):
found = True
grid[attemptindex] = sample
active.append(sample)
if found == False:
del active[randindex]
for printout in range(len(grid)):
print("<" + str(active[printout][0]) + "," + str(active[printout][1]) + "," + str(active[printout][2]) + ">")
print(len(grid))
我的代码似乎永远运行。因此,我尝试在 while 循环的最后添加一个 print(len(active)) 。令人惊讶的是,我想我发现了这个错误,因为活动列表的长度一直在增加!(它应该与网格的长度相同)我认为问题是由 active.append() 引起的,但我无法弄清楚问题出在哪里,因为代码实际上与 90% 相同希夫曼先生制作的一款。我不想搭便车,但我已经一次又一次地检查,同时一次又一次地纠正这段代码:(。不过,我不知道错误在哪里。(为什么 active[] 继续追加! ?) 感谢您宝贵的时间。