任务:需要拟合圆周与轴线位置
难点:三维圆拟合与检测都很复杂,没有方便可用的成熟方案,最小二乘法既无法处理高维情况,也会受异常点干扰,RANSAC的检测迭代次数过多,选点的随机性过大
基本思路:通过深度信息过滤干扰平面,使用RANSAC算法检测最大平面作为圆筒端面,在该端面上随机选取一个点作为初始化圆心,以最小化所有点到圆心的距离差作为优化目标,求解最优化问题,得到圆心和半径,结合端面法向量可以求出轴线方程。
相关数据:深度信息过滤后的点云数据下载
file = '3'
file_before = 'duanmian/'
pcd = o3d.io.read_point_cloud(file_before + file + '/point_cloud_00000.ply')
points = np.array(pcd.points)
colors = np.zeros(np.array(pcd.points).shape[0])
pcd.colors = o3d.utility.Vector3dVector(np.zeros(np.array(pcd.colors).shape))
o3d.visualization.draw_geometries([pcd])
pcd = pcd.uniform_down_sample(every_k_points = 20)
o3d.visualization.draw_geometries([pcd])
o3d.io.write_point_cloud(file_before + file + '/old.ply', pcd)
plane_model, inliers = pcd.segment_plane(distance_threshold=3 * 1e-3,
ransac_n=3,
num_iterations=1000)
[a, b, c, d] = plane_model
print(f"Plane equation: {a:.2f}x + {b:.2f}y + {c:.2f}z + {d:.2f} = 0")
inlier_cloud = pcd.select_by_index(inliers)
outlier_cloud = pcd.select_by_index(inliers, invert=True)
#统计滤波
# nb_neighbors:最近k个点 std_ratio:基于标准差的阈值,越小滤除点越多
cl,ind = inlier_cloud.remove_statistical_outlier(nb_neighbors=3, std_ratio=1)
inlier_cloud = inlier_cloud.select_by_index(ind)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
outlier_cloud2 = inlier_cloud.select_by_index(ind, invert=True)
#半径滤波
# nb_points:基于球体内包含点数量的阈值 radius:半径
cl,ind = inlier_cloud.remove_radius_outlier(nb_points=3, radius = 1.0)
inlier_cloud = inlier_cloud.select_by_index(ind)
model(points: torch.tensor, cir: torch.tensor, line: torch.tensor)
def model(points: torch.tensor, cir: torch.tensor, line: torch.tensor):
'''
功能:
根据圆心和待拟合点计算损失
输入:
待拟合点,待优化圆心(二维),端面平面方程
输出:
圆心,半径,损失
'''
line = line.float()
points = points.float()
cir = cir.float()
#计算圆心三维坐标
cir_z = torch.matmul(cir, line[0:2].T) + line[-1]
cir_z = cir_z / (1e-10 - line[2])
cir_z = cir_z.unsqueeze(0)
cir = torch.cat([cir, cir_z], 0)
#计算半径矩阵和损失
#损失一定程度上表示每个点到圆心的距离的差距
points = points - cir
points = torch.matmul(points, points.T)
points = torch.diag(points)
n_all = points.shape[0]
r_all = torch.sum(points) / (n_all ** 1)
e = 0
for i in range(1,4):
n = int(n_all / i)
r = torch.sum(points[:n]) / (n ** 1)
e += ((r - r_all) ** 2 )
return cir, r_all ** 0.5, e
points_2 = np.array(inlier_cloud.points) #* 100
cir =torch.from_numpy(points_2[0][0:2])
cir.requires_grad = True
points_2 = torch.from_numpy(points_2)
line = torch.Tensor(np.array([a, b, c, d]))
learning_rate_o = 1e-3
learning_rate_2 = 1e-2
learning_rate_3 = 1
learning_rate_4 = 8
repect_n = 0
repect = 0
epoch = 0
jingdu = 1e-28
epoch_max = 5 * 1e5
print('-------开始学习---------')
while(True):
epoch += 1
if cir.grad is not None:
#梯度归零
cir.grad.zero_()
#前向传播
_, r, l = model(points_2, cir, line)
#反向传播
l.backward()
if cir.grad is None:
#梯度爆炸就及时退出
print('++++++++++++')
print('epoch:', epoch)
print('a:', cir)
print('grad:', cir.grad)
print('r:', r)
break
#分段学习率
if l < 100:
learning_rate = learning_rate_2
if l < 45:
learning_rate = learning_rate_3
if l < 0.2:
learning_rate = learning_rate_4
else:learning_rate = learning_rate_3
else:learning_rate = learning_rate_2
else:
learning_rate = learning_rate_o
with torch.no_grad():
cir -= learning_rate * cir.grad
if epoch % 5e3 == 0:
print('------------------')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
if l < jingdu:
print('精度足够,停止学习')
break
if epoch > epoch_max:
break
if l == repect:
repect_n += 1
else:
repect = l
repect_n = 0
if repect_n > 15:
print('达到收敛停止学习')
break
print('*****************************')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
cir, r, l = model(points_2, cir, line)
print('圆心坐标:(', cir, '),半径:', r.item())
see = np.row_stack([np.array(inlier_cloud.points), cir.detach().numpy()])
inlier_cloud.points = o3d.utility.Vector3dVector(see)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
#空间圆可视化https://www.doc88.com/p-813917521845.html
def get_points_of_circle_3d(line, cir, r):
'''
已知圆心、半径、圆所在平面方程,计算该圆的散点和轴线散点
'''
A, B, C, D = line
#取平面上不贡献三个点,组成不共线两个向量
p1 = np.array([0, 0, -1 * D / C])
p2 = np.array([1, 0, (-1 * D - A) / C])
p3 = np.array([0, 1, (-1 * D - B) / C])
u1 = p1 - p2
u2 = p1 - p3
#法向量
n = np.cross(u1, u2)
n_cir = [cir + n * x for x in np.arange(-0.1, 0.1, 0.001)]
#print(n)
#圆平面建立坐标系
v = np.cross(u1, n)
#转为单位向量
u = u1 / (np.dot(u1, u1.T) ** 0.5)
v = v / (np.dot(v, v.T) ** 0.5)
#根据参数方程生成圆的散点
import math
p_cir = [cir + u * r * math.cos(x) + v * r * math.sin(x) for x in np.arange(0, 2*3.15 ,0.05)]
return np.array(p_cir), np.array(n_cir)
p_cir, n_cir = get_points_of_circle_3d(line.detach().numpy(), cir.detach().numpy(), r.detach().numpy())
see = np.row_stack([cir.detach().numpy(), p_cir, n_cir])
cir_cloud = o3d.geometry.PointCloud()
cir_cloud.points = o3d.utility.Vector3dVector(see)
cir_cloud.paint_uniform_color([0, 1.0, 0])
o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud, outlier_cloud, cir_cloud])
new_pcd = o3d.geometry.PointCloud()
new_pcd.points = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.points),
np.array(outlier_cloud.points),
np.array(outlier_cloud2.points),
np.array(cir_cloud.points)]))
new_pcd.colors = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.colors),
np.array(outlier_cloud.colors),
np.array(outlier_cloud2.colors),
np.array(cir_cloud.colors)]))
o3d.io.write_point_cloud(file_before + file + '/new.ply', new_pcd)
import open3d as o3d
import numpy as np
from numpy.linalg import det
import random
import torch
def model(points: torch.tensor, cir: torch.tensor, line: torch.tensor):
'''
功能:
根据圆心和待拟合点计算损失
输入:
待拟合点,待优化圆心(二维),端面平面方程
输出:
圆心,半径,损失
'''
line = line.float()
points = points.float()
cir = cir.float()
#计算圆心三维坐标
cir_z = torch.matmul(cir, line[0:2].T) + line[-1]
cir_z = cir_z / (1e-10 - line[2])
cir_z = cir_z.unsqueeze(0)
cir = torch.cat([cir, cir_z], 0)
#计算半径矩阵和损失
#损失一定程度上表示每个点到圆心的距离的差距
points = points - cir
points = torch.matmul(points, points.T)
points = torch.diag(points)
n_all = points.shape[0]
r_all = torch.sum(points) / (n_all ** 1)
e = 0
for i in range(1,4):
n = int(n_all / i)
r = torch.sum(points[:n]) / (n ** 1)
e += ((r - r_all) ** 2 )
return cir, r_all ** 0.5, e
file = '3'
file_before = 'duanmian/'
pcd = o3d.io.read_point_cloud(file_before + file + '/point_cloud_00000.ply')
#pcd = pcd.voxel_down_sample(voxel_size=5e-3)
points = np.array(pcd.points)
colors = np.zeros(np.array(pcd.points).shape[0])
pcd.colors = o3d.utility.Vector3dVector(np.zeros(np.array(pcd.colors).shape))
#o3d.visualization.draw_geometries([pcd])
pcd = pcd.uniform_down_sample(every_k_points = 20)
#o3d.visualization.draw_geometries([pcd])
o3d.io.write_point_cloud(file_before + file + '/old.ply', pcd)
plane_model, inliers = pcd.segment_plane(distance_threshold=3 * 1e-3,
ransac_n=3,
num_iterations=1000)
[a, b, c, d] = plane_model
print(f"Plane equation: {a:.2f}x + {b:.2f}y + {c:.2f}z + {d:.2f} = 0")
inlier_cloud = pcd.select_by_index(inliers)
outlier_cloud = pcd.select_by_index(inliers, invert=True)
print('------开始滤波------')
#参考https://blog.csdn.net/skycol/article/details/127429843
#统计滤波
# nb_neighbors:最近k个点 std_ratio:基于标准差的阈值,越小滤除点越多
cl,ind = inlier_cloud.remove_statistical_outlier(nb_neighbors=3, std_ratio=1)
inlier_cloud = inlier_cloud.select_by_index(ind)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
outlier_cloud2 = inlier_cloud.select_by_index(ind, invert=True)
#半径滤波
# nb_points:基于球体内包含点数量的阈值 radius:半径
#cl,ind = inlier_cloud.remove_radius_outlier(nb_points=3, radius = 1.0)
#inlier_cloud = inlier_cloud.select_by_index(ind)
# o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud, outlier_cloud2])
points_2 = np.array(inlier_cloud.points) #* 100
cir =torch.from_numpy(points_2[0][0:2])#选取第一个点作为初始化圆心
cir.requires_grad = True
points_2 = torch.from_numpy(points_2)
line = torch.Tensor(np.array([a, b, c, d]))
learning_rate_o = 1e-3
learning_rate_2 = 1e-2
learning_rate_3 = 1
learning_rate_4 = 8
repect_n = 0
repect = 0
epoch = 0
jingdu = 1e-28
epoch_max = 5 * 1e5
print('-------开始学习---------')
while(True):
epoch += 1
if cir.grad is not None:
#梯度归零
cir.grad.zero_()
#前向传播
_, r, l = model(points_2, cir, line)
#反向传播
l.backward()
if cir.grad is None:
#梯度爆炸就及时退出
print('++++++++++++')
print('epoch:', epoch)
print('a:', cir)
print('grad:', cir.grad)
print('r:', r)
break
#分段学习率
if l < 100:
learning_rate = learning_rate_2
if l < 45:
learning_rate = learning_rate_3
if l < 0.2:
learning_rate = learning_rate_4
else:learning_rate = learning_rate_3
else:learning_rate = learning_rate_2
else:
learning_rate = learning_rate_o
with torch.no_grad():
cir -= learning_rate * cir.grad
if epoch % 5e3 == 0:
print('------------------')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
if l < jingdu:
print('精度足够,停止学习')
break
if epoch > epoch_max:
break
if l == repect:
repect_n += 1
else:
repect = l
repect_n = 0
if repect_n > 15:
print('达到收敛停止学习')
break
print('*****************************')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
cir, r, l = model(points_2, cir, line)
print('圆心坐标:(', cir, '),半径:', r.item())
see = np.row_stack([np.array(inlier_cloud.points), cir.detach().numpy()])
inlier_cloud.points = o3d.utility.Vector3dVector(see)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
#o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud])
#空间圆可视化https://www.doc88.com/p-813917521845.html
def get_points_of_circle_3d(line, cir, r):
'''
已知圆心、半径、圆所在平面方程,计算该圆的散点和轴线散点
'''
A, B, C, D = line
#取平面上不贡献三个点,组成不共线两个向量
p1 = np.array([0, 0, -1 * D / C])
p2 = np.array([1, 0, (-1 * D - A) / C])
p3 = np.array([0, 1, (-1 * D - B) / C])
u1 = p1 - p2
u2 = p1 - p3
#法向量
n = np.cross(u1, u2)
n_cir = [cir + n * x for x in np.arange(-0.1, 0.1, 0.001)]
#print(n)
#圆平面建立坐标系
v = np.cross(u1, n)
#转为单位向量
u = u1 / (np.dot(u1, u1.T) ** 0.5)
v = v / (np.dot(v, v.T) ** 0.5)
#根据参数方程生成圆的散点
import math
p_cir = [cir + u * r * math.cos(x) + v * r * math.sin(x) for x in np.arange(0, 2*3.15 ,0.05)]
return np.array(p_cir), np.array(n_cir)
p_cir, n_cir = get_points_of_circle_3d(line.detach().numpy(), cir.detach().numpy(), r.detach().numpy())
see = np.row_stack([cir.detach().numpy(), p_cir, n_cir])
cir_cloud = o3d.geometry.PointCloud()
cir_cloud.points = o3d.utility.Vector3dVector(see)
cir_cloud.paint_uniform_color([0, 1.0, 0])
o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud, outlier_cloud, cir_cloud])
new_pcd = o3d.geometry.PointCloud()
new_pcd.points = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.points),
np.array(outlier_cloud.points),
np.array(outlier_cloud2.points),
np.array(cir_cloud.points)]))
new_pcd.colors = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.colors),
np.array(outlier_cloud.colors),
np.array(outlier_cloud2.colors),
np.array(cir_cloud.colors)]))
o3d.io.write_point_cloud(file_before + file + '/new.ply', new_pcd)
import open3d as o3d
import numpy as np
from numpy.linalg import det
import random
import torch
import sys
def points_3_to_2(points_3, flag):
#投影公式参考https://blog.csdn.net/weixin_39849839/article/details/108313284
A = flag[0]
B = flag[1]
C = flag[2]
D = flag[3]
change = [[B ** 2 + C ** 2, A ** 2 + C ** 2, A ** 2 + B **2],
[-1 * A * B, -1 * B * A, -1 * A * C],
[-1 * A * C, -1 * B * C, -1 * C * B],
[-1 * A * D, -1 * B * D, -1 * C * D ]]
change = np.array(change)
change /= (A ** 2 + B ** 2 + C ** 2)
points_3 = np.column_stack([points_3, np.ones(points_3.shape[0])])
#print(points_3)
points_2 = np.dot(points_3, change)
return points_2
def model(points: torch.tensor, cir: torch.tensor, line: torch.tensor):
'''
功能:
输入:
'''
line = line.float()
points = points.float()
cir = cir.float()
cir_z = torch.matmul(cir, line[0:2].T) + line[-1]
cir_z = cir_z / (1e-10 - line[2])
cir_z = cir_z.unsqueeze(0)
cir = torch.cat([cir, cir_z], 0)
points = points - cir
points = torch.matmul(points, points.T)
points = torch.diag(points)
n_all = points.shape[0]
r_all = torch.sum(points) / (n_all ** 1)
e = 0
for i in range(1,4):
n = int(n_all / i)
r = torch.sum(points[:n]) / (n ** 1)
e += ((r - r_all) ** 2 )
return cir, r_all ** 0.5, e
with open("config/config.txt","r",encoding="utf-8") as f:
file_path = f.readline().split(":")[1]
ply_name = file_path.split("/")[-1]
file_path = '/'.join(file_path.split("/")[:-1])
log = open(file_path + "/log.txt",'w',encoding="utf-8")
sys.stdout = log
print("file_path:",file_path)
print("ply_name:", ply_name)
pcd = o3d.io.read_point_cloud(file_path + '/' + ply_name)
#pcd = pcd.voxel_down_sample(voxel_size=5e-3)
points = np.array(pcd.points)
colors = np.zeros(np.array(pcd.points).shape[0])
pcd.colors = o3d.utility.Vector3dVector(np.zeros(np.array(pcd.colors).shape))
#o3d.visualization.draw_geometries([pcd])
pcd = pcd.uniform_down_sample(every_k_points = 20)
#o3d.visualization.draw_geometries([pcd])
o3d.io.write_point_cloud(file_path + '/old.ply', pcd)
plane_model, inliers = pcd.segment_plane(distance_threshold=3 * 1e-3,
ransac_n=3,
num_iterations=1000)
[a, b, c, d] = plane_model
print(f"Plane equation: {a:.2f}x + {b:.2f}y + {c:.2f}z + {d:.2f} = 0")
inlier_cloud = pcd.select_by_index(inliers)
print('------开始滤波------')
#参考https://blog.csdn.net/skycol/article/details/127429843
#统计滤波
# nb_neighbors:最近k个点 std_ratio:基于标准差的阈值,越小滤除点越多
cl,ind = inlier_cloud.remove_statistical_outlier(nb_neighbors=3, std_ratio=1)
inlier_cloud = inlier_cloud.select_by_index(ind)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
outlier_cloud2 = inlier_cloud.select_by_index(ind, invert=True)
#半径滤波
# nb_points:基于球体内包含点数量的阈值 radius:半径
#cl,ind = inlier_cloud.remove_radius_outlier(nb_points=3, radius = 1.0)
#inlier_cloud = inlier_cloud.select_by_index(ind)
outlier_cloud = pcd.select_by_index(inliers, invert=True)
# o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud, outlier_cloud2])
points_2 = np.array(inlier_cloud.points) #* 100
cir =torch.from_numpy(points_2[0][0:2])
cir.requires_grad = True
points_2 = torch.from_numpy(points_2)
line = torch.Tensor(np.array([a, b, c, d]))
learning_rate_o = 1e-3
learning_rate_2 = 1e-2
learning_rate_3 = 1
learning_rate_4 = 8
repect_n = 0
repect = 0
epoch = 0
jingdu = 1e-28
epoch_max = 5 * 1e5
print('-------开始学习---------')
while(True):
epoch += 1
if cir.grad is not None:
cir.grad.zero_()
_, r, l = model(points_2, cir, line)
l.backward()
if cir.grad is None:
print('++++++++++++')
print('epoch:', epoch)
print('a:', cir)
print('grad:', cir.grad)
print('r:', r)
#print('rate:',learning_rate)
#print('loss:', l)
break
#print(a.grad)
#learning_rate = 0.1 ** ((len(str(cir.grad[0])) / 2))
if l < 100:
learning_rate = learning_rate_2
if l < 45:
learning_rate = learning_rate_3
if l < 0.2:
learning_rate = learning_rate_4
else:learning_rate = learning_rate_3
else:learning_rate = learning_rate_2
else:
learning_rate = learning_rate_o
with torch.no_grad():
"""
if torch.isnan(a.grad).any():
print('------------------')
print('epoch:',epoch)
print('a:', a)
print('grad:', a.grad)
print('rate:',learning_rate)
print('loss:', l)
break
"""
cir -= learning_rate * cir.grad
#print(a - learning_rate * a.grad)
if epoch % 5e3 == 0:
print('------------------')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
if l < jingdu:
print('精度足够,停止学习')
break
if epoch > epoch_max:
break
if l == repect:
repect_n += 1
else:
repect = l
repect_n = 0
if repect_n > 15:
print('达到收敛停止学习')
break
print('*****************************')
print('epoch:',epoch)
print('a:', cir)
print('grad:', cir.grad)
print('rate:',learning_rate)
print('loss:', l)
print('r:', r.item())
cir, r, l = model(points_2, cir, line)
print('圆心坐标:(', cir, '),半径:', r.item())
see = np.row_stack([np.array(inlier_cloud.points), cir.detach().numpy()])
inlier_cloud.points = o3d.utility.Vector3dVector(see)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
#o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud])
#空间圆可视化https://www.doc88.com/p-813917521845.html
def get_points_of_circle_3d(line, cir, r):
'''
'''
A, B, C, D = line
#取平面上不贡献三个点,组成不共线两个向量
p1 = np.array([0, 0, -1 * D / C])
p2 = np.array([1, 0, (-1 * D - A) / C])
p3 = np.array([0, 1, (-1 * D - B) / C])
u1 = p1 - p2
u2 = p1 - p3
#法向量
n = np.cross(u1, u2)
n_cir = [cir + n * x for x in np.arange(-0.1, 0.1, 0.001)]
#print(n)
#圆平面建立坐标系
v = np.cross(u1, n)
#转为单位向量
u = u1 / (np.dot(u1, u1.T) ** 0.5)
v = v / (np.dot(v, v.T) ** 0.5)
#根据参数方程生成圆的散点
import math
p_cir = [cir + u * r * math.cos(x) + v * r * math.sin(x) for x in np.arange(0, 2*3.15 ,0.05)]
return np.array(p_cir), np.array(n_cir)
p_cir, n_cir = get_points_of_circle_3d(line.detach().numpy(), cir.detach().numpy(), r.detach().numpy())
see = np.row_stack([cir.detach().numpy(), p_cir, n_cir])
cir_cloud = o3d.geometry.PointCloud()
cir_cloud.points = o3d.utility.Vector3dVector(see)
cir_cloud.paint_uniform_color([0, 1.0, 0])
o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud, outlier_cloud, cir_cloud])
new_pcd = o3d.geometry.PointCloud()
new_pcd.points = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.points),
np.array(outlier_cloud.points),
np.array(outlier_cloud2.points),
np.array(cir_cloud.points)]))
new_pcd.colors = o3d.utility.Vector3dVector(np.row_stack([np.array(inlier_cloud.colors),
np.array(outlier_cloud.colors),
np.array(outlier_cloud2.colors),
np.array(cir_cloud.colors)]))
o3d.io.write_point_cloud(file_path + '/new.ply', new_pcd)
log.close()
Pyinstaller -F -w -i duanmian/FindCircle.ico FindCircle.py