3D拟合测量水杯半径

news2025/7/26 22:14:40

1,目的。

测量水杯的半径

如图所示:
在这里插入图片描述


2,原理。

3D 点云对象 进行圆柱体拟合,获取拟合后的半径。


3,注意事项。

  1. 在Halcon中使用fit_primitives_object_model_3d进行圆柱体拟合时,输出的primitive_parameter包含以下7个参数:
    1. 参数构成
      ​ - 轴线方向向量
      3个数值:(a, b, c)
      描述圆柱体轴线的空间方向,满足归一化条件 a² + b² + c² = 1
      . - 轴线上基准点坐标
      3个数值:(x0, y0, z0)
      表示轴线上的一个参考点,通常靠近点云的重心
      . - ‌圆柱半径
      1个数值:R
      单位为点云的原始坐标系单位(如毫米)

    2. 数学验证

      圆柱表面点需满足方程:
      ( x − x p ) 2 + ( y − y p ) 2 + ( z − z p ) 2 − [ ( x − x p ) a + ( y − y p ) b + ( z − z p ) c ] 2 = R \sqrt{(x-x_p)^2+(y-y_p)^2+(z-z_p)^2-[(x-x_p)a+(y-y_p)b+(z-z_p)c]^2}=R (xxp)2+(yyp)2+(zzp)2[(xxp)a+(yyp)b+(zzp)c]2 =R
      其中(x_p, y_p, z_p)为轴线上任意一点


  1. 在Halcon中使用fit_primitives_object_model_3d对三维点云进行球体拟合时,输出的primitive_parameter包含以下4个关键参数:

    1. 参数构成
      球心坐标
      包含3个数值:(x, y, z)
      表示拟合球体的中心点在三维空间中的位置
      球体半径
      单个数值:R
      单位为点云的原始坐标系单位(如毫米或米)

    2. 参数验证方法
      数学验证‌:球面点需满足方程 ( x − x 0 ) 2 + ( y − y 0 ) 2 + ( z − z 0 ) 2 = R 2 (x-x_0)^2+(y-y_0)^2+(z-z_0)^2=R^2 (xx0)2+(yy0)2+(zz0)2=R2
      其中(x_0, y_0, z_0)为球心坐标

    3. 与其他几何体的区别

    • 圆柱体‌:输出7个参数(轴线方向、基准点、半径)68
    • 平面‌:输出4个参数(法向量、偏移量)

4,代码。

*参考案例库:fit_primitives_object_model_3d.hdev

* ***********************************************************************
* 将输入的 2.5D 图像拟合成圆柱体,并计算显示拟合后的圆柱体半径

* 2.5D 图像:在常规二维图像基础上叠加‌**高度或深度数据**‌,形成半三维表达(如灰度值隐含高度信息或直接存储深度通道)*
* ***********************************************************************
dev_update_off ()
dev_close_window ()
* Input: 2.5D image
* tif图像:是HALCON支持的‌高精度图像格式‌,专门用于存储包含复杂信息的图像数据,如3D深度图、工业相机采集的点云数据等*
* 一般由结构光相机,3D扫描仪提供
read_image (XYZ, '3d_machine_vision/segmentation/3d_primitives_xyz_02.tif')
dev_open_window_fit_image (XYZ, 0, 0, -1, -1, WindowHandle)
set_display_font (WindowHandle, 16, 'mono', 'true', 'false')
* Access to (x-, y-, z-)coordinates
access_channel (XYZ, X, 1)
access_channel (XYZ, Y, 2)

*  提取第三通道作为深度
access_channel (XYZ, Z, 3)
* Segment cylinder manually
threshold (Z, Region, 0.0, 0.83)

reduce_domain (X, Region, XTmp)

dev_display (Z)
dev_set_draw ('margin')
dev_set_line_width (2)
dev_set_color ('green')
dev_display (Region)
disp_message (WindowHandle, 'Segment cylinder with a simple threshold', 'window', 12, 12, 'black', 'true')
disp_continue_message (WindowHandle, 'black', 'true')
stop ()
* 
xyz_to_object_model_3d (XTmp, Y, Z, ObjectModel3DID)

* 指定 3D 对象的原始类型,拟合算法,输出类型
ParFitting := ['primitive_type','fitting_algorithm','output_xyz_mapping']
ValFitting := ['cylinder','least_squares_huber','true']
* 对 原始对象进行 3D 拟合

fit_primitives_object_model_3d (ObjectModel3DID, ParFitting, ValFitting, ObjectModel3DOutID)
* Clear the object model that is no longer used
clear_object_model_3d (ObjectModel3DID)
* Display the result of the fitting
dev_display (Z)
dev_set_draw ('fill')
dev_display_fitting_results (RegionCylinder, RegionSphere, RegionPlane, RegionNone, ObjectModel3DOutID, WindowHandle, [])
dev_update_on ()
clear_object_model_3d (ObjectModel3DOutID)

本地函数:dev_display_fitting_results

* This procedure displays the results of a fitting operation.
* Additionally to the primitive regions, the radius of
* cylinders and spheres is displayed.
* The colors can be specified with the parameter Colors
* in the following manner:
* Color[0] is used for objects which could not be fitted
* Color[1] is used for cylinders:用于圆柱体
* Color[2] is used for spheres:用于球体
* Color[3] is used for planes:用于平面
* 
if (Colors == [])
    Colors := ['dim gray','forest green','red','slate blue']
endif
NumNone := 0
* 圆柱体数量
NumCylinders := 0
* 球体数量
NumSpheres := 0
* 平面数量
NumPlanes := 0
for Index := 0 to |ObjectModel3DOutID| - 1 by 1
    object_model_3d_to_xyz (XTmp, YTmp, ZTmp, ObjectModel3DOutID[Index], 'from_xyz_map', [], [])
    get_domain (XTmp, DomainTmp)
    * 是否存在原始数据
    get_object_model_3d_params (ObjectModel3DOutID[Index], 'has_primitive_data', ParamValue)
    if (ParamValue == 'true')
        get_object_model_3d_params (ObjectModel3DOutID[Index], 'primitive_parameter', GenParamValuesP)
        get_object_model_3d_params (ObjectModel3DOutID[Index], 'primitive_type', ParamValue)
        if (ParamValue == 'cylinder')
            if (NumCylinders == 0)
                *-1:表示复制从索引开始剩余的所有对象,注意这里是浅复制
                copy_obj (DomainTmp, RegionCylinder, 1, -1)
                * 从拟合的原始数据中获取拟合的圆柱体半径值
                RadiusCylinder := GenParamValuesP[6]
            else
                concat_obj (RegionCylinder, DomainTmp, RegionCylinder)
                RadiusCylinder := [RadiusCylinder,GenParamValuesP[6]]
            endif
            NumCylinders := NumCylinders + 1
        elseif (ParamValue == 'sphere')
            * 球体情况
            if (NumSpheres == 0)
                copy_obj (DomainTmp, RegionSphere, 1, -1)
                RadiusSphere := GenParamValuesP[3]
            else
                concat_obj (RegionSphere, DomainTmp, RegionSphere)
                RadiusSphere := [RadiusSphere,GenParamValuesP[3]]
            endif
            NumSpheres := NumSpheres + 1
        else
            * 平面情况
            if (NumPlanes == 0)
                copy_obj (DomainTmp, RegionPlane, 1, -1)
            else
                concat_obj (RegionPlane, DomainTmp, RegionPlane)
            endif
            NumPlanes := NumPlanes + 1
        endif
    else
        if (NumNone == 0)
            copy_obj (DomainTmp, RegionNone, 1, -1)
        else
            concat_obj (RegionNone, DomainTmp, RegionNone)
        endif
        NumNone := NumNone + 1
    endif
endfor
* 
NumColors := |Colors|
if (NumNone > 0)
    dev_set_color (Colors[0 % NumColors])
    dev_display (RegionNone)
endif
if (NumCylinders > 0)
    dev_set_color (Colors[1 % NumColors])
    dev_display (RegionCylinder)
endif
if (NumSpheres > 0)
    dev_set_color (Colors[2 % NumColors])
    dev_display (RegionSphere)
endif
if (NumPlanes > 0)
    dev_set_color (Colors[3 % NumColors])
    dev_display (RegionPlane)
endif
disp_message (WindowHandle, '3D Fitting', 'window', -1, -1, 'black', 'true')
Message := 'Cylinders: ' + NumCylinders
Message[1] := 'Spheres: ' + NumSpheres
Message[2] := 'Planes: ' + NumPlanes
Message[3] := 'Undefined: ' + NumNone
disp_message (WindowHandle, Message, 'window', 40, 12, 'black', 'true')
if (NumCylinders > 0)
    * Display radius for each cylinder
    area_center (RegionCylinder, Area, Row, Column)
    for Index := 0 to NumCylinders - 1 by 1
        Radius := int(RadiusCylinder[Index] * 1000)
        disp_message (WindowHandle, 'Cylinder\nr = ' + Radius + ' mm', 'image', Row[Index], Column[Index], 'white', 'false')
    endfor
endif
if (NumSpheres > 0)
    * Display radius for each cylinder sphere
    area_center (RegionSphere, Area, Row, Column)
    for Index := 0 to NumSpheres - 1 by 1
        Radius := int(RadiusSphere[Index] * 1000)
        disp_message (WindowHandle, 'Sphere\nr = ' + Radius + ' mm', 'image', Row[Index], Column[Index], 'white', 'false')
    endfor
endif
return ()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2397700.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python训练打卡Day38

Dataset和Dataloader类 知识点回顾: Dataset类的__getitem__和__len__方法(本质是python的特殊方法)Dataloader类minist手写数据集的了解 在遇到大规模数据集时,显存常常无法一次性存储所有数据,所以需要使用分批训练的…

Selenium基础操作方法详解

Selenium基础操作方法详解:从零开始编写自动化脚本(附完整代码) 引言 Selenium是自动化测试和网页操作的利器,但对于新手来说,掌握基础操作是成功的第一步。本文将手把手教你使用Selenium完成浏览器初始化、元素定位、…

简单三步FastAdmin 开源框架的安装

简单三步FastAdmin 开源框架的安装 第一步:新建站点1,在宝塔面板中,创建一个新的站点,并填写项目域名。 第二步:上传框架1,框架下载2,上传解压缩 第三步:配置并安装1,进入…

RISC-V 开发板 MUSE Pi Pro 搭建 Spacengine AI模型部署环境

视频讲解: RISC-V 开发板 MUSE Pi Pro 搭建 Spacengine AI模型部署环境 Spacengine 是由 进迭时空 研发的一套 AI 算法模型部署工具,可以方便的帮助用户部署自己的模型在端侧, 环境部署的方式,官方提供了两种方式: do…

【Unity】AudioSource超过MaxDistance还是能听见

unity版本:2022.3.51f1c1 将SpatialBlend拉到1即可 或者这里改到0 Hearing audio outside max distance - #11 by wderstine - Questions & Answers - Unity Discussions

Qt 读取和写入 INI 格式的配置文件

Qt 读取和写入 INI 格式的配置文件 前言:INI 配置文件在 Qt 开发中的重要性基础夯实:INI 文件结构与 QSettings 核心概念1. INI 文件的基本结构2. QSettings 类概述3. 初始化 QSettings 对象4. 基本读写操作5. 高级操作技巧5.1 处理数组和列表5.2 检查键…

Spring:从青铜到王者,你的Java修炼手册

一、Spring家族宇宙:原来你是这样的框架(青铜段位) 1.1 Spring的"前世今生":从泡面到满汉全席 ​​2002年的泡面哲学​​:Rod Johnson在厨房煮泡面时突然顿悟:"Java开发为什么不能像泡面一…

Qt creator 设计页面控件认识与了解

记录一下 Qt 中的认识与了解: 在 Qt 中,这些功能属于 Qt Designer 的组件过滤系统,旨在帮助开发者在对象浏览器中快速定位和使用不同类型的控件和组件。以下是每个功能的详细讲解: ‌Layouts(布局)‌&…

NVIDIA Mellanox BlueField-2 DPU(Data Processing Unit)智能网卡的调试和使用

专有名词 OOB: BMC: BFB: EMMC: 关键词解释eMMCEmbedded Multi-Media Card——把 NAND 闪存颗粒与控制器封装在一起的板载存储件,类似手机里的“内置储存” .deb:文件是​​Debian软件包格式​​的专…

Tomcat- AJP协议文件读取/命令执行漏洞(幽灵猫复现)详细步骤

一、漏洞描述 Apache Tomcat是由Apache软件基金会属下Jakarta项目开发的Servlet容器.默认情况下,Apache Tomcat会开启AJP连接器,方便与其他Web服务器通过AJP协议进行交互.但Apache Tomcat在AJP协议的实现上存在漏洞,导致攻击者可以通过发送恶意的AJP请求,可以读取或者包含Web应…

B1、进度汇报(— 25/05/31)

本文档汇总了各成员在 2025 年 5 月 11 日 ~ 5 月 31 日完成的工作。我们遇到了进度问题(收工后需反思): 本学期第十四周(05/19 ~ 05/25)有相当多课程需要提交实验结果或上台展示。本学期第十六周(06/02 ~…

Flutter实现不规则瀑布流布局拖拽重排序

因为业务,所以需要用flutter去实现一种不规则图形的瀑布流,但是同时需要支持拖拽并重新排序。效果类似如下。 查询过现有的插件,要么是仅支持同样大小的组件进行排序,要么就是动画效果不是很满意,有点死板,…

【第4章 图像与视频】4.1 图像的绘制

文章目录 前言在 Canvas 之中绘制图像drawImage() 方法的用法 前言 drawImage() 方法可以将一幅图像的整体或某个部分绘制到 canvas 内的任何位置上,并且允许开发者在绘制过程中对图像进行缩放。也可以将图像绘制在离屏 canvas 中,这样的话就可以对图像…

G25-05-31Rust开源项目日报 Top10

根据Github Trendings的统计,今日(2025-05-31统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量Rust项目10TypeScript项目1Pake: 利用 Rust 轻松构建轻量级多端桌面应用 创建周期:491 天开发语言:Rust协议类型:MIT LicenseStar数量:2095…

window安装nginx

步骤1:下载Nginx for Windows​ 访问Nginx官网下载页面:https://nginx.org/en/download.html 在​​Stable version​​(稳定版)下找到Windows版本,点击下载.zip文件(如 nginx-1.28.0.zip) 步…

ArcGIS Pro裁剪影像

方法1: 工具箱中,数据管理工具-栅格-栅格处理-裁剪栅格 注意:勾选上使用输入要素裁剪几何 方法2: 按掩膜提取工具裁剪栅格:在工具箱中,Spatial Analyst工具-提取分析-按掩膜提取

[智能算法]蚁群算法原理与TSP问题示例

目录 ​编辑 一、生物行为启发的智能优化算法 1.1 自然界的群体智能现象 1.2 人工蚁群算法核心思想 二、算法在组合优化中的应用演进 2.1 经典TSP问题建模 2.2 算法流程优化 三、TSP问题实战:Python实现与可视化 3.1 算法核心类设计 3.2 参数敏感性实验 3.3 可视化…

【Go语言】Fyne GUI 库使用指南 (面向有经验开发者)

引言 Fyne 是一个使用 Go 语言编写的、易于使用的跨平台 GUI 工具包和应用程序 API。它旨在通过单一代码库构建在桌面和移动设备上运行的应用程序。本文档面向有一定 Go 语言开发经验的开发者,将详细介绍 Fyne 最新版的核心功能,包括基础组件、布局系统…

openssl 怎么生成吊销列表

mkdir test cd test # 根据 /usr/lib/ssl/openssl.cnf 配置文件中目录结构可知有个demoCA目录,目录下有各种文件。 mkdir ./demoCA ./demoCA/newcerts ./demoCA/private sudo chmod 777 -R ./demoCA/ echo 01 > ./demoCA/serial touch ./demoCA/index.txt # /usr…

springboot-响应接收与ioc容器控制反转、Di依赖注入

1.想将服务器中的数据返回给客户端,需要在controller类上加注解:ResponseBody; 这个注解其实在前面已经使用过,RestController其实就包含两个注解: Controller ResponseBody 返回值如果是实体对象/集合,将会转换为j…