除了 CGAL python 绑定之外,python 中是否有 3 维的“alpha 形状”函数?
或者,有没有办法将下面的示例扩展到 3D?
2D 示例:在 matplotlib 中围绕散点图中的数据点绘制平滑多边形
我目前正在使用这个ConvexHull示例计算体积,但出于我的目的,由于“凸”约束,体积被夸大了。
谢谢,
除了 CGAL python 绑定之外,python 中是否有 3 维的“alpha 形状”函数?
或者,有没有办法将下面的示例扩展到 3D?
2D 示例:在 matplotlib 中围绕散点图中的数据点绘制平滑多边形
我目前正在使用这个ConvexHull示例计算体积,但出于我的目的,由于“凸”约束,体积被夸大了。
谢谢,
我写了一些代码来查找 alpha 形状表面。我希望这有帮助。
from scipy.spatial import Delaunay
import numpy as np
from collections import defaultdict
def alpha_shape_3D(pos, alpha):
"""
Compute the alpha shape (concave hull) of a set of 3D points.
Parameters:
pos - np.array of shape (n,3) points.
alpha - alpha value.
return
outer surface vertex indices, edge indices, and triangle indices
"""
tetra = Delaunay(pos)
# Find radius of the circumsphere.
# By definition, radius of the sphere fitting inside the tetrahedral needs
# to be smaller than alpha value
# http://mathworld.wolfram.com/Circumsphere.html
tetrapos = np.take(pos,tetra.vertices,axis=0)
normsq = np.sum(tetrapos**2,axis=2)[:,:,None]
ones = np.ones((tetrapos.shape[0],tetrapos.shape[1],1))
a = np.linalg.det(np.concatenate((tetrapos,ones),axis=2))
Dx = np.linalg.det(np.concatenate((normsq,tetrapos[:,:,[1,2]],ones),axis=2))
Dy = -np.linalg.det(np.concatenate((normsq,tetrapos[:,:,[0,2]],ones),axis=2))
Dz = np.linalg.det(np.concatenate((normsq,tetrapos[:,:,[0,1]],ones),axis=2))
c = np.linalg.det(np.concatenate((normsq,tetrapos),axis=2))
r = np.sqrt(Dx**2+Dy**2+Dz**2-4*a*c)/(2*np.abs(a))
# Find tetrahedrals
tetras = tetra.vertices[r<alpha,:]
# triangles
TriComb = np.array([(0, 1, 2), (0, 1, 3), (0, 2, 3), (1, 2, 3)])
Triangles = tetras[:,TriComb].reshape(-1,3)
Triangles = np.sort(Triangles,axis=1)
# Remove triangles that occurs twice, because they are within shapes
TrianglesDict = defaultdict(int)
for tri in Triangles:TrianglesDict[tuple(tri)] += 1
Triangles=np.array([tri for tri in TrianglesDict if TrianglesDict[tri] ==1])
#edges
EdgeComb=np.array([(0, 1), (0, 2), (1, 2)])
Edges=Triangles[:,EdgeComb].reshape(-1,2)
Edges=np.sort(Edges,axis=1)
Edges=np.unique(Edges,axis=0)
Vertices = np.unique(Edges)
return Vertices,Edges,Triangles
您正在寻找“凹壳”。行进立方体算法可用于找到这样的船体。您可以在此处找到完整示例。
限制:如果您的数据来自体积数据集,或者如果您有可以轻松转换为体积数据集(类似体素)的点云,则此方法效果很好。这可以使用一组密集的点相对容易地完成,例如,使用像scipy cKDTree这样的空间索引器,但如果你有一个稀疏的点云,你最终可能会挠头以获得好的结果。