OPEN3D Document-()Geometry-Point cloud

发布于:2022-12-19 ⋅ 阅读:(1614) ⋅ 点赞:(2)

Introduction(介绍)

Open3d是一个支持解决3D数据快速处理软件开发的一个开源库。Open3D 前端在 C++ 和 Python 中公开了一组精心挑选的数据结构和算法。后端被高度优化,以及采用并行化处理运行。我们欢迎来自开源社区的贡献。

Open3d包含的核心特点:

  • 3D数据结构
  • 3D数据处理算法
  • 场景重建
  • 表面对准
  • 3D可视化
  • 基于物理的渲染(Physically based rendering (PBR))
  • 支持PyTorch和TensorFlow的3D机器学习
  • 对核心的3D运算支持GPU加速
  • 允许C++和Python变成
  • 等等

Python quick start (快速开始)

构建Pip 或conda,需支持Ubuntu 18.04+, macOS 10.15+ and Windows 10 (64-bit)平台上的Python 3.6-3.9。

# Install
pip install open3d

# Verify installation
python -c "import open3d as o3d; print(o3d.__version__)"

# Python API
python -c "import open3d as o3d; \
           mesh = o3d.geometry.TriangleMesh.create_sphere(); \
           mesh.compute_vertex_normals(); \
           o3d.visualization.draw(mesh, raw_mode=True)"

# Open3D CLI
open3d example visualization/draw

C++ quick start (C++快速开始-未翻译)

讨论渠道

GitHub Issue: bug reports, feature requests, etc.

Forum: 关于Open3D使用的讨论。

Discord Chat: 在线聊天, 讨论,以及和他使用者或开发者合作。

论文引用(Latex)

@article{Zhou2018,
    author    = {Qian-Yi Zhou and Jaesik Park and Vladlen Koltun},
    title     = {{Open3D}: {A} Modern Library for {3D} Data Processing},
    journal   = {arXiv:1801.09847},
    year      = {2018},
}

Getting start(让我们开始吧【Python】)

python

Open3D的python包是通过PyPI发布的。

支持的Python版本:3.7、3.8、3.9、3.10

支持的操作系统:Ubuntu 18.04+、macOS 10.15+、Windows 10 (64-bit)

install: 一般来说,我们建议使用虚拟环境或conda环境。否则,根据配置的不同,Python 3可能需要使用pip3,或者需要使用–user选项来避免权限问题。比如说。

pip3 install open3d
# or
pip install --user open3d
# or
python3 -m pip install --user open3d

运行Open3d的教程

将复制一套完整的Python教程和数据集合,以演示Open3D Python接口的使用。
https://github.com/isl-org
https://github.com/isl-org/Open3D
python教程在上述链接的/examples/python中, 里面有一些ply格式的数据)(不是Master版本,其他版本),(搞不懂,我就想下载一个ply数据就找不到,外过网站的那个兔子一直下不下来,国内的都是要钱的,很无语)

C++ (未翻译)

Build from source(从源头开始简历建立)

系统要求:

C++14 compiler:
	Ubuntu 18.04+: GCC 5+, Clang 7+
	macOS 10.15+: XCode 8.0+
	Windows 10 (64-bit): Visual Studio 2019+
CMake: 3.19+
	Ubuntu (18.04 / 20.04):
			Install with apt-get: see official APT repository
			Install with snap: sudo snap install cmake --classic
			Install with pip (run inside a Python virtualenv): pip install cmake
macOS: Install with Homebrew: brew install cmake
Windows: Download from: [CMake download page](https://cmake.org/download/)、
CUDA 10.1+ :可选择的,Open3D 通过 Linux 上的 CUDA 支持对越来越多的操作进行 GPU 加速。我们建议使用CUDA11.0, 以便与最近的GPU和可选的外部依赖(如Tensorflow或PyTorch)保持最佳兼容性。
如果有需要请查询官方文档安装CUDA toolkit   【https://docs.nvidia.com/cuda/cuda-installation-guide-linux/index.html】

使用git工具克隆Open3D

git clone https://github.com/isl-org/Open3D

Ubuntu和macOS安装依赖

# Only needed for Ubuntu
util/install_deps_ubuntu.sh

Python环境的设置

激活python的虚拟环境(virtualenv)或Conda环境。

在C++项目中链接Open3D (未翻译)

建立文档(未翻译)

Open3D-ML

在这里插入图片描述
Open3D-ML是Open3D为3D机器学习任务所实现的的扩展。他是在Open3D核心库上建立的,并且扩展她成为3D数据处理的机器学习工具。

ARM支持(未翻译)

Docker(未翻译)

下面为TUTORIAL

几何学(Geometry)

Point cloud(点云)

本教程演示了点云的基本用法。

点云可视化

  1. 点云的读取和可视化
import open3d as o3d
import numpy as np
 
print("Load a ply point cloud, print it, and render it")
 
# 从文件中读取点云
# 下载地址为
# https://github.com/isl-org/open3d_downloads/releases/download/20220201-data/fragment.ply
# 链接:https://pan.baidu.com/s/1V3anL9caeDe6bWgHsrEpTg 
# 提取码:5u7b 
pcd = o3d.io.read_point_cloud('./fragment.ply')
print(pcd)
print(np.asarray(pcd.points))
# 可视化点云。使用鼠标/触控板从不同的视点查看几何图形。
o3d.visualization.draw_geometries([pcd],
                                  zoom=0.3412,
                                  front=[0.4257, -0.2125, -0.8795],
                                  lookat=[2.6172, 2.0475, 1.532],
                                  up=[-0.0694, -0.9768, 0.2024])

在这里插入图片描述

==read_point_cloud:==从一个文件中读取点云。它尝试根据扩展名解码文件。对于列表的文件类型,请参考File IO.
==draw_geometries:==可视化点云,使用鼠标或触摸板从不同的角度查看不同的点云。
It looks like a dense surface, 上图看起来像一个深度表面图(这半句翻译不准确,dense surface),但是,但它实际上是点云渲染成的表面。GUI支持各种键盘功能。例如,"-"键减少了点的大小(surfels)。
eg:在这里插入图片描述
在可视化的图上按键“h”按键,在控制台会出现帮助(关于快捷键的技巧)更多可视化和自定义可视化的技巧参考 Visualization and Customized visualization., 此外macOS中的快捷键可能不能用,此时试试将python替换成pythonw。

[Open3D INFO]   -- 鼠标控制 --
[Open3D INFO]     鼠标左键 + 拖 = 旋转
[Open3D INFO]     Ctrl + 鼠标左键 + 拖拽 = 平移.
[Open3D INFO]     鼠标滑轮 =  放大/缩小.
[Open3D INFO]     Shift + 鼠标左键 + 拖拽 : 滚动

[Open3D INFO]   -- 键盘控制 --
[Open3D INFO]     [,] 方括号  : 增加/减少视野。
[Open3D INFO]     R            : 重置视角。
[Open3D INFO]     Ctrl/Cmd + C : 将当前视图状态复制到剪贴板。
[Open3D INFO]     Ctrl/Cmd + V : 将当前视图状态粘贴。

[Open3D INFO]   -- 普通控制 --
[Open3D INFO]     Q, Esc       : 退出窗口
[Open3D INFO]     H            : 打印帮助信息
[Open3D INFO]     P, PrtScn    : 捕获当前图像并以png和json保存信息
[Open3D INFO]     D            : 同P
[Open3D INFO]     O            : 对当前的渲染设置进行捕捉,以json的形式存储。
[Open3D INFO]     Alt + Enter  : 在全屏和窗口模式之间进行切换。

[Open3D INFO]   -- 渲染模式控制--
[Open3D INFO]     L            : Turn on/off lighting(曝光??).
[Open3D INFO]     +/-          : 增加/减少 点的大小.
[Open3D INFO]     Ctrl + +/-   :增加/减少几何体的宽度::LineSet.(没有用)
[Open3D INFO]     N            : 打开和关闭点云正常的渲染
[Open3D INFO]     S            : 在网格平坦阴影和平滑阴影之间切换。
[Open3D INFO]     W            : 打开/关闭网状线框。(没什么用)
[Open3D INFO]     B            : Turn on/off back face rendering.
[Open3D INFO]     I            : Turn on/off image zoom in interpolation.
[Open3D INFO]     T            : Toggle among image render:
[Open3D INFO]                    no stretch / keep ratio / freely stretch.

[Open3D INFO]   -- 颜色控制 --
[Open3D INFO]                    0 - 默认渲染的点云
[Open3D INFO]                    1 - Render point color.
[Open3D INFO]                    2 - x coordinate as color.
[Open3D INFO]                    3 - y coordinate as color.
[Open3D INFO]                    4 - z coordinate as color.
[Open3D INFO]                    9 - normal as color.
[Open3D INFO]     Ctrl + 0..4,9: Set mesh color option.
[Open3D INFO]                    0 - Default behavior, render uniform gray color.
[Open3D INFO]                    1 - Render point color.
[Open3D INFO]                    2 - x coordinate as color.
[Open3D INFO]                    3 - y coordinate as color.
[Open3D INFO]                    4 - z coordinate as color.
[Open3D INFO]                    9 - normal as color.
[Open3D INFO]     Shift + 0..4 : Color map options.
[Open3D INFO]                    0 - Gray scale color.
[Open3D INFO]                    1 - JET color map.
[Open3D INFO]                    2 - SUMMER color map.
[Open3D INFO]                    3 - WINTER color map.
[Open3D INFO]                    4 - HOT color map.
[Open3D INFO] 
Process finished with exit code 0
  1. 体素(Voxel)下采样,(二维的是像素,可能三维的就是体素了)
    体素下采样使用规则的体素网格,从输入的点云中创建一个均匀下采样的点云。它经常被用作许多点云处理任务的预处理步骤。该算法分两步操作。
    a.点云被归入体素
    b.每个占用的体素通过对内部所有点进行平均来生成一个点。
import open3d as o3d
import numpy as np

print("Load a ply point cloud, print it, and render it")

# 从文件中读取点云
pcd = o3d.io.read_point_cloud(r'D:\ros\data\open3D_test\data\fragment.ply')
print(pcd)
print(np.asarray(pcd.points))
print("Downsample the point cloud with a voxel of 0.05")
downpcd = pcd.voxel_down_sample(voxel_size=0.05)

# 可视化点云。使用鼠标/触控板从不同的视点查看几何图形。
o3d.visualization.draw_geometries([downpcd],
                                  zoom=0.3412,
                                  front=[0.4257, -0.2125, -0.8795],
                                  lookat=[2.6172, 2.0475, 1.532],
                                  up=[-0.0694, -0.9768, 0.2024])

在这里插入图片描述

  1. 顶点法线估计(Vertex normal estimation)
    按键盘的N建。或者以下代码,键 - 和 + 可用于控制法线的长度。
import open3d as o3d
import numpy as np

print("Load a ply point cloud, print it, and render it")

# 从文件中读取点云
pcd = o3d.io.read_point_cloud(r'D:\ros\data\open3D_test\data\fragment.ply')
print(pcd)
print(np.asarray(pcd.points))
print("Downsample the point cloud with a voxel of 0.05")
downpcd = pcd.voxel_down_sample(voxel_size=0.05)

print("Recompute the normal of the downsampled point cloud")
downpcd.estimate_normals(search_param=o3d.geometry.KDTreeSearchParamHybrid(radius=0.1, max_nn=30))
o3d.visualization.draw_geometries([downpcd],
                                  zoom=0.3412,
                                  front=[0.4257, -0.2125, -0.8795],
                                  lookat=[2.6172, 2.0475, 1.532],
                                  up=[-0.0694, -0.9768, 0.2024],
                                  point_show_normal=True)

estimate_normals: 计算每个点的法线。该函数找到相邻的点,并利用协方差分析计算相邻点的主轴。
该函数将KDTreeSearchParamHybrid类的一个实例作为参数。两个关键参数radius = 0.1和max_nn = 30指定了搜索半径和最大近邻。它有10cm的搜索半径,并且只考虑最多30个邻居以节省计算时间。

Note:
协方差分析算法产生了两个相反的方向作为候选法线。在不知道几何体的全局结构的情况下,两个方向都可能是正确的。这就是所谓的法线方向问题。如果原法线存在,Open3D会尝试将法线的方向与原法线对齐。否则,Open3D会做一个随机猜测。如果方向是一个问题,就需要进一步调用orient_normals_to_align_with_direction和orient_normals_towards_camera_location等方向函数。

  1. 输出估计的顶点法线(Access estimated vertex normal)
    估计的法线向量可以从downpcd的法线变量中检索
print("Print a normal vector of the 0th point")
print(downpcd.normals[0])

# 法线向量转化为array
print("Print the normal vectors of the first 10 points")
print(np.asarray(downpcd.normals)[:10, :])
  1. 点云裁剪(Crop point cloud)
    数据下载地址上方有,或点我或者点我
import open3d as o3d
import numpy as np

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
###################################################
vol = o3d.visualization.read_selection_polygon_volume("./data/cropped.json")
chair = vol.crop_point_cloud(pcd)
###################################################
o3d.visualization.draw_geometries([chair],
                                  zoom=0.7,
                                  front=[0.5439, -0.2333, -0.8060],
                                  lookat=[2.4615, 2.1331, 1.338],
                                  up=[-0.1781, -0.9708, 0.1608])

read_selection_polygon_volume:读取指定多边形选择区域的json文件
vol.crop_point_cloud(pcd):过滤出点云
在这里插入图片描述

  1. 点云添色(Paint point cloud)
import open3d as o3d
import numpy as np

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
vol = o3d.visualization.read_selection_polygon_volume("./data/cropped.json")
chair = vol.crop_point_cloud(pcd)
###################################################
chair.paint_uniform_color([1, 0.706, 0])
###################################################
o3d.visualization.draw_geometries([chair],
                                  zoom=0.7,
                                  front=[0.5439, -0.2333, -0.8060],
                                  lookat=[2.4615, 2.1331, 1.338],
                                  up=[-0.1781, -0.9708, 0.1608])

paint_uniform_color: 将所有点云绘制成一个标准颜色。RGB颜色的范围是【0. 1】
在这里插入图片描述
12. 点云距离(Point cloud distance)
compute_point_cloud_distance方法计算源点云到目标点云之间的距离等。它为源点云中的每个点计算到目标点云中最近点的距离。
在下面的例子中,我们使用这个函数来计算两个点云之间的差异。请注意,这种方法也可以用来计算两个点云之间的Chamfer距离。

import open3d as o3d
import numpy as np

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
vol = o3d.visualization.read_selection_polygon_volume("./data/cropped.json")
chair = vol.crop_point_cloud(pcd)
chair.paint_uniform_color([1, 0.706, 0])
###################################################
dists = pcd.compute_point_cloud_distance(chair)
dists = np.asarray(dists)
ind = np.where(dists > 0.01)[0]
pcd_without_chair = pcd.select_by_index(ind)
###################################################
o3d.visualization.draw_geometries([pcd_without_chair],
                                  zoom=0.7,
                                  front=[0.5439, -0.2333, -0.8060],
                                  lookat=[2.4615, 2.1331, 1.338],
                                  up=[-0.1781, -0.9708, 0.1608])

在这里插入图片描述
14. 边界体积 (Bounding volumes)
PointCloud 几何类型与 Open3D 中的所有其他几何类型一样具有边界体积。目前,Open3D 实现了一个 AxisAlignedBoundingBox 和一个 OrientedBoundingBox,它们也可用于裁剪几何图形。

import open3d as o3d
import numpy as np

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
vol = o3d.visualization.read_selection_polygon_volume("./data/cropped.json")
chair = vol.crop_point_cloud(pcd)
chair.paint_uniform_color([1, 0.706, 0])

# 求解距离,根据距离索引点云
dists = pcd.compute_point_cloud_distance(chair)
dists = np.asarray(dists)
ind = np.where(dists > 0.01)[0]
pcd_without_chair = pcd.select_by_index(ind)


###################################################
# 体积
aabb = chair.get_axis_aligned_bounding_box()
aabb.color = (1, 0, 0)
obb = chair.get_oriented_bounding_box()
obb.color = (0, 1, 0)
###################################################

o3d.visualization.draw_geometries([chair, aabb, obb],
                                  zoom=0.7,
                                  front=[0.5439, -0.2333, -0.8060],
                                  lookat=[2.4615, 2.1331, 1.338],
                                  up=[-0.1781, -0.9708, 0.1608])

在这里插入图片描述
16. 凸包(Convex hull)
点云的凸包是包含所有点的最小凸集。Open3D 包含计算点云凸包的方法 compute_convex_hull。该实现基于 Qhull。
在下面的示例代码中,我们首先从一个网格中对点云进行采样,并计算出凸壳,然后以三角形网格的形式返回。然后,我们将凸体可视化为一个红色的LineSet。

import open3d as o3d
import numpy as np

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
mesh = o3d.io.read_triangle_mesh("./data/BunnyMesh.ply")
mesh.compute_vertex_normals()
vol = o3d.visualization.read_selection_polygon_volume("./data/cropped.json")
chair = vol.crop_point_cloud(pcd)
chair.paint_uniform_color([1, 0.706, 0])

# 求解距离,根据距离索引点云
dists = pcd.compute_point_cloud_distance(chair)
dists = np.asarray(dists)
ind = np.where(dists > 0.01)[0]
pcd_without_chair = pcd.select_by_index(ind)



# 体积
aabb = chair.get_axis_aligned_bounding_box()
aabb.color = (1, 0, 0)
obb = chair.get_oriented_bounding_box()
obb.color = (0, 1, 0)


###################################################
# 凸包
pcl = mesh.sample_points_poisson_disk(number_of_points=2000)
hull, _ = pcl.compute_convex_hull()
hull_ls = o3d.geometry.LineSet.create_from_triangle_mesh(hull)
hull_ls.paint_uniform_color((1, 0, 0))
o3d.visualization.draw_geometries([pcl, hull_ls])
###################################################

在这里插入图片描述
17. DBSCAN 聚类(DBSCAN clustering)
给定一个来自例如深度传感器的点云,我们想把局部的点云聚类在一起。为了这个目的,我们可以使用聚类算法。Open3D实现了DBSCAN[Ester1996],这是一种基于密度的聚类算法。该算法在cluster_dbscan中实现,需要两个参数:eps定义了集群中与邻居的距离,min_points定义了形成一个集群所需的最小点数。该函数返回标签,其中标签-1表示噪声.

import open3d as o3d
import numpy as np
import matplotlib as plt

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
with o3d.utility.VerbosityContextManager(
        o3d.utility.VerbosityLevel.Debug) as cm:
    labels = np.array(
        pcd.cluster_dbscan(eps=0.02, min_points=10, print_progress=True))

max_label = labels.max()
print(f"point cloud has {max_label + 1} clusters")
colors = plt.cm.get_cmap("tab20")(labels / (max_label if max_label > 0 else 1))
colors[labels < 0] = 0
pcd.colors = o3d.utility.Vector3dVector(colors[:, :3])
o3d.visualization.draw_geometries([pcd],
                                  zoom=0.455,
                                  front=[-0.4999, -0.1659, -0.8499],
                                  lookat=[2.1813, 2.0619, 2.0999],
                                  up=[0.1204, -0.9852, 0.1215])

在这里插入图片描述
18. 平面分割(plane segmentation)
Open3D也支持使用RANSAC对点云中的几何基元进行分割。为了找到点云中支持度最大的平面,我们可以使用 segment_plane。

该方法有三个参数:
distance_threshold定义了一个点与一个估计的平面的最大距离,以便被认为是一个离群点;r
ansac_n定义了为估计一个平面而随机采样的点的数量;
num_iterations定义了一个随机平面被采样和验证的频率。

然后,该函数返回平面为(a,b,c,d),这样对于平面上的每个点(x,y,z),我们有ax+by+cz+d=0。

import open3d as o3d
import numpy as np
import matplotlib as plt

# 从文件中读取点云
print("Load a polygon volume and use it to crop the original point cloud")
pcd = o3d.io.read_point_cloud("./data/fragment.ply")
plane_model, inliers = pcd.segment_plane(distance_threshold=0.01,
                                         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)
inlier_cloud.paint_uniform_color([1.0, 0, 0])
outlier_cloud = pcd.select_by_index(inliers, invert=True)
o3d.visualization.draw_geometries([inlier_cloud, outlier_cloud],
                                  zoom=0.8,
                                  front=[-0.4999, -0.1659, -0.8499],
                                  lookat=[2.1813, 2.0619, 2.0999],
                                  up=[0.1204, -0.9852, 0.1215])

在这里插入图片描述
19. 隐藏点移除(Hidden point removal)
想象一下,你想从一个给定的视点渲染一个点云,但是背景中的点因为没有被其他点遮挡而漏到了前景中。为了这个目的,我们可以应用一个隐藏点去除算法。在Open3D中,实现了[Katz2007]的方法,在没有表面重建或法线估计的情况下,从一个给定的视图中接近点云的可见性。

import open3d as o3d
import numpy as np
import matplotlib as plt

print("Convert mesh to a point cloud and estimate dimensions")
mesh = o3d.io.read_triangle_mesh("./data/ArmadilloMesh.ply")
mesh.compute_vertex_normals()

pcd = mesh.sample_points_poisson_disk(5000)
diameter = np.linalg.norm(
    np.asarray(pcd.get_max_bound()) - np.asarray(pcd.get_min_bound()))
o3d.visualization.draw_geometries([pcd])

#################################################################
print("Define parameters used for hidden_point_removal")
camera = [0, 0, diameter]
radius = diameter * 100

print("Get all points that are visible from given view point")
_, pt_map = pcd.hidden_point_removal(camera, radius)

print("Visualize result")
pcd = pcd.select_by_index(pt_map)
o3d.visualization.draw_geometries([pcd])

在这里插入图片描述
#################################################################
在这里插入图片描述