深入理解halcon相机标定[通俗易懂]

深入理解halcon相机标定[通俗易懂]目录相机标定简介深度说明1、相机标定参数介绍2、标定板详细介绍问题1:halcon是否只能使用halcon专用的标定板?问题2:halcon标定板如何生成?问题3:halcon标定板如何摆放,拍照数量有无限制?标定步骤1、设置相机内部初始值使用set_calib_data_cam_param算子设置相机内部初始值畸变类型选择与参数确定技巧2、标定板初始化3、创建标定数据模型4、获取标定图片5、使…

大家好,又见面了,我是你们的朋友全栈君。

  • 博主写作不容易,还是需要您鼓励
  • 万水千山总是情 , 先点个赞行不行

最近有的人私信我,说看了我的《HALCON相机标定相机内参相机外参》https://blog.csdn.net/cashmood/article/details/100089295一文后,还是有不懂的地方,这里给大家做一个深入的教程。

相机标定简介

首先镜头有畸变,也就是说照出的图像与实际不符产生了形变。即使工业镜头也是有千分之几的畸变率的。
上个图告诉大家畸变
在这里插入图片描述
这个图里,第一个图就是我们相机下的真实的形状,后边两个就是照出来有畸变的图片。

其次镜头与相机无论你的机械结构精度多高,也不容易或者说没办法将相机安装的特别正,那相机安装不正也是会导致误差的。大家想知道具体数学模型的话可以搜一下相机标定的理论方面的知识,我侧重怎么做。
标定就是把上述两个东西转化成正常的。
无论是在图像测量或者机器视觉应用中,相机参数的标定都是非常关键的环节,其标定结果的精度及算法的稳定性直接影响相机工作产生结果的准确性。

深度说明

1、相机标定参数介绍

在这里插入图片描述
内参:确定摄像机从三维空间到二维空间的投影关系。
针孔相机(FA镜头相机)模型为6个参数(f,kSx,Sy,Cx,Cy);远心镜头相机模型为5个参数(f,Sx,Sy,Cx,Cy);线阵相机为11个参数(f,k,Sx,Sy,Cx,Cy,Width,Highth,Vx,Vy,Vz)。
其中:
f为焦距;
k表示径向畸变量级。如果k为负值,畸变为桶形畸变,如果为正值,那么畸变为枕形畸变。
Sx,Sy是缩放比例因子。对于针孔摄像机(FA镜头)表示图像传感器水平和垂直方向上相邻像素之间的距离,初始值与真实值越接近计算速度越快。对于远心摄像机模型,表示像素在世界坐标系中的尺寸。
Cx,Cy是图像的主点,对于针孔相机,这个点是投影中心在成像平面上的垂直投影,同时也是径向畸变的中心。对于远心摄像机模型,只表示畸变的中心。
Vx,Vy,Vz:线阵相机必须与被拍摄物体之间有相对移动才能拍摄到一幅有用的图像。这是运动向量。Sx,Sy对于线阵相机是相邻像元的水平和垂直距离。

2、标定板详细介绍

问题1:halcon是否只能使用halcon专用的标定板?

halcon提供了简便、精准的标定算子与标定助手,这在实际使用中极大地方便了使用者
在halcon中有两种标定方式:
halcon自带例程中出现的,用halcon定义的标定板,如下图:
在这里插入图片描述
用户自定义标定板,用户可以制作任何形状、形式的标定板,如下图:
在这里插入图片描述
所以,halcon并非只能使用专用标定板,也可以使用自定义标定板就可以进行标定。

问题2:halcon标定板如何生成?

使用gen_caltab算子来制作一个标定板
gen_caltab( : : XNum, YNum, MarkDist, DiameterRatio, CalPlateDescr, CalPlatePSFile : )
XNum:每行黑色标志圆点的数量
YNum:每列黑色标志圆点的数量
MarkDist:两个就近黑色圆点中心之间的距离,单位是m
DiameterRatio:黑色圆点直径与两圆点中心距离的比值
CalPlateDescr:标定板描述文件的文件路径
CalPlatePSFile :标定板图像文件路径,可以用ps打开
示例:
gen_caltab( 7, 7, 0.1, 0.5, ‘caltab.descr’, ‘caltab.ps’)
大家可以自己思考一下每个参数的含义
行数:7
列数:7
黑色圆点半径:0.05m
圆点中心间距:0.1m

问题3:halcon标定板如何摆放,拍照数量有无限制?

并非拍照标定数量越多,越能取得高的精度,halcon建议拍摄数量在9-16张,并且对摆放位置做了建议。标定板占标定视野的1/3-1/4,对于标定板成像灰度值应大于128,以便顺利提取。如下图所示:
在这里插入图片描述

标定步骤

注意只介绍面阵相机,其他的相机和面阵道理想同。

1、设置相机内部初始值

使用set_calib_data_cam_param 算子设置相机内部初始值

set_calib_data_cam_param( : : CalibDataID, CameraIdx, CameraType, CameraParam : )
CalibDataID:标定句柄
CameraIdx:相机序号
CameraType:相机模型种类;面阵相机Division畸变模型’area_scan_division’;polynomial畸变模型’area_scan_polynomial’。。。。。。
CameraParam :与相机模型种类相对应的参数;面阵相机Division畸变模型’area_scan_division’([Focus, Kappa, Sx, Sy, Cx, Cy, ImageWidth, ImageHeight]);polynomial畸变模型’area_scan_polynomial’([Focus, K1, K2, K3, P1, P2, Sx, Sy, Cx, Cy, ImageWidth, ImageHeight])。。。。。。

相机模型种类与相机参数对应表:
在这里插入图片描述

畸变类型选择与参数确定技巧

畸变模型选择
division畸变模型只适用于精度不高的,标定图片数量较少的情况:
polynomial畸变模型对径向畸变、切向畸变都进行矫正,精度较高,花费时间较长。
参数确定技巧
Focus代表焦距,按照我们镜头参数进行填写,远心镜头填写0

Kappa为畸变大小,因为在标定之前,所以默认填写0

Sx, Sy像元的宽高填写相机的像元尺寸。可以查相机手册,或者咨询相机厂家。
Cx, Cy填写图像的中心坐标
ImageWidth, ImageHeight填写图像的宽高

2、标定板初始化

使用算子 set_calib_data_calib_object
例如:
CaltabDescr := ‘caltab_100mm.descr’
set_calib_data_calib_object (CalibDataID, 0, CaltabDescr)
这个比较简单,大家看帮助就可以懂,就不进行介绍了

3、创建标定数据模型

使用算子 create_calib_data( : : CalibSetup, NumCameras, NumCalibObjects : CalibDataID)
CalibSetup:创建的内容
NumCameras:相机个数
NumCalibObjects :标定项目数
CalibDataID:标定句柄
例如:
create_calib_data (‘calibration_object’, 1, 1, CalibDataID)

4、获取标定图片

标定板为正方形,尺寸大小为要照射区域宽度的1/3,相机拍摄或者读入9-16张图片,拍摄图片时标定板尽量覆盖整个视场;标定板圆点的直径不能小于10个像素。

5、使用图像进行相机内参标定,得到结果

方式一:
使用标定图像,直接用halcon全自动,进行标定
find_calib_object (Image, CalibDataID, CameraIndex, 0, PoseIndex, [], [])
得到标定数据
get_calib_data (CalibDataID, ‘camera’, 0, ‘type’, CameraType)

方式二:
1、寻找标定板区域,确定圆心,将结果加载到组元中
find_caltab(Image : CalPlate : CalPlateDescr, SizeGauss, MarkThresh, MinDiamMarks : )
find_marks_and_pose(Image, CalPlateRegion : : CalPlateDescr, StartCamParam, StartThresh, DeltaThresh, MinThresh, Alpha, MinContLength, MaxDiamMarks : RCoord, CCoord, StartPose)
set_calib_data_observ_points( : : CalibDataID, CameraIdx, CalibObjIdx, CalibObjPoseIdx, Row, Column, Index, Pose : )
原理:首先find_caltab算子对图像高斯滤波(核大小为SizeGauss),接着阈值分割(阈值大小为MarkThresh)将标定板的区域找出来;find_marks_and_pose算子对区域中的圆进行分割,找到圆的个数,周长,坐标等应该和标定板描述文件中的一致,否则会自动调整StartThresh,使得StartThresh按照DeltaThresh步长减小到MinDiamMarks ,直到找到准确的圆心。
2、进行标定
calibrate_cameras( : : CalibDataID : Error)
返回平均投影误差Error

6、使用图像进行相机外参标定,得到结果

摄像机外参:决定摄像机坐标系与世界坐标系之间相对位置关系
其中Pw为世界坐标,Pc为摄像机坐标,T=(Tx,Ty,Tz)是平移向量,R=(a,b,r)是旋转矩阵,分别是绕摄像机坐标系Z轴旋转角度为r;绕摄像机坐标系Y轴旋转角度为b;绕摄像机坐标系X轴旋转角度为a。6个参数组成摄像机外参(a,b,r,Tx,Ty,Tz)
并且满足下式:
Pc=R*Pw+T

具体步骤:
set_calib_data_cam_param( : : CalibDataID, CameraIdx, CameraType, CameraParam : )
set_calib_data_calib_object( : : CalibDataID, CalibObjIdx, CalibObjDescr : )
find_calib_object(Image : : CalibDataID, CameraIdx, CalibObjIdx, CalibObjPoseIdx, GenParamName, GenParamValue : )
get_calib_data_observ_contours( : Contours : CalibDataID, ContourName, CameraIdx, CalibObjIdx, CalibObjPoseIdx : )
get_calib_data_observ_points( : : CalibDataID, CameraIdx, CalibObjIdx, CalibObjPoseIdx : Row, Column, Index, Pose)
set_origin_pose( : : PoseIn, DX, DY, DZ : PoseNewOrigin)
在这里插入图片描述

Halcon相机标定代码与解析

**创建标定板
gen_caltab(7,7,0.008,0.5,'48_48mm.descr','48_48mm.ps')
           
*=======标定内参
dev_close_window ()
dev_open_window (0, 0, 652, 494, 'black', WindowHandle)
dev_update_off ()
dev_set_draw ('margin')
dev_set_line_width (3)
OpSystem := environment('OS')
set_display_font (WindowHandle, 14, 'mono', 'true', 'false')
           
*标定相机
StartCamPar := [0.0,0.0,0.0000299,0.0000299,4896/2,3264/2,4896,3264]
create_calib_data ('calibration_object', 1, 1, CalibDataID)
set_calib_data_cam_param (CalibDataID, 0, 'area_scan_telecentric_division', StartCamPar)
set_calib_data_calib_object (CalibDataID, 0, '48_48mm.descr')
           

for index := 1 to 13 by 1
    read_image (Image, '标定20/' + index + '.png')
    get_image_size(Image, Width, Height)
    dev_display (Image)
    find_calib_object (Image, CalibDataID, 0, 0, index, [], [])
    get_calib_data_observ_contours (Caltab, CalibDataID, 'caltab', 0, 0, index)
    dev_set_color ('green')
    dev_display (Caltab)
endfor
           
calibrate_cameras (CalibDataID, Error)
get_calib_data (CalibDataID, 'camera', 0, 'params', CamParam)
get_calib_data (CalibDataID, 'calib_obj_pose', [0,1], 'pose', PoseCalib)
           
*输出计算的相机内参
write_cam_par (CamParam, 'camera_parameters.dat')
           
           
Message:= '相机内参已经写入文件中'
disp_message (WindowHandle, Message, 'window', 12, 12, 'red', 'false')
clear_calib_data (CalibDataID)
stop()
           
           
*=====标定外参
dev_set_draw ('margin')
dev_set_line_width (1)
set_display_font (WindowHandle, 14, 'mono', 'true', 'false')
           
*从文件中读取内参 存储文件:camera_parameters.dat
try
    read_cam_par ('camera_parameters.dat', CamParam)
catch (Exception)
    stop ()
endtry
           
*开始计算
open_file('data.csv','output', FileHandle)
fwrite_string(FileHandle,'Dis_pix*0.0299204,Dis_m*1000,Distance')
fnew_line (FileHandle)
close_file(FileHandle)
           
*选择一张作为标定作为最终标定位姿(任意一张都可以)
index:=1
read_image (Image,'标定20/'+index+'.png')
dev_display (Image)
CaltabName := '48_48mm.descr'
create_calib_data ('calibration_object', 1, 1, CalibDataID)

set_calib_data_cam_param (CalibDataID, 0, 'area_scan_telecentric_division', CamParam)
set_calib_data_calib_object (CalibDataID, 0, CaltabName)
find_calib_object (Image, CalibDataID, 0, 0, 1, [], [])
get_calib_data_observ_contours (Caltab, CalibDataID, 'caltab', 0, 0, 1)
get_calib_data_observ_points (CalibDataID, 0, 0, 1, RCoord, CCoord, Index, PoseForCalibrationPlate)
dev_set_color ('green')
dev_display (Caltab)
dev_set_color ('red')
disp_caltab (WindowHandle, CaltabName, CamParam, PoseForCalibrationPlate, 1)
dev_set_line_width (1)
disp_circle (WindowHandle, RCoord, CCoord, gen_tuple_const(|RCoord|,1.5))
* caltab_points (CaltabName, X, Y, Z)
* calibrate_cameras (CalibDataID, Error)
* To take the thickness of the calibration plate into account, the z-value
* of the origin given by the camera pose has to be translated by the
* thickness of the calibration plate.
* Deactivate the following line if you do not want to add the correction.
set_origin_pose (PoseForCalibrationPlate, 0, 0, 0, PoseCalib)
* disp_continue_message (WindowHandle, 'black', 'true')
* stop ()
           
*像素距离
distance_pp(RCoord[0],CCoord[0],RCoord[48],CCoord[48], Dis_pix)
*像素直接转换mm然后计算
pix2mm(RCoord, CCoord,CamParam[2],CamParam[3],newCol,newRow)
distance_pp(newRow[0],newCol[0],newRow[48],newCol[48], Dis_m)
*用同一个世界坐标系来计算
image_points_to_world_plane(CamParam, PoseCalib,[RCoord[0],RCoord[48]], [CCoord[0],CCoord[48]], 'mm', X1, Y1)
distance_pp(Y1[0],X1[0],Y1[1],X1[1],Distance)
           
*输出计算结果比较
open_file('data.csv','append', FileHandle)
fwrite_string(FileHandle, Dis_pix*0.0299+','+Dis_m*1000+','+Distance+'\n')
close_file(FileHandle)    
           
Message:= '计算完毕'
disp_message (WindowHandle, Message, 'window', 12, 12, 'red', 'false')
stop()

此文档由http://www.ihalcon.com/read-7598.html分享整理得到。

  • 博主简介:
  • 工业自动化上位机软件工程师、机器视觉算法工程师、运动控制算法工程师。目前从业于智能制造自动化行业。 博主邮箱:2296776525@qq.com
  • 帮忙点个赞吧。哈哈。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/140443.html原文链接:https://javaforall.net

(0)
上一篇 2022年5月27日 下午11:40
下一篇 2022年5月28日 上午6:00


相关推荐

  • 2、认识python

    1,python历史。宏观上:python2 与 python3 区别:python2 源码不标准,混乱,重复代码太多,python3 统一 标准,去除重复代码。2,python的环境。编译型:一次性将所有程序编译成二进制文件。缺点:开发效率低,不能跨平台。优点:运行速度快。:C,C++等等。解释型:当程序执行时,一行一行的解释。优点:开发效率高,可以跨…

    2021年11月30日
    47
  • Java Bean 简介及其应用

    Java Bean 简介及其应用Bean的中文含义是“豆子”,顾名思义JavaBean是一段Java小程序。JavaBean实际上是指一种特殊的Java类,它通常用来实现一些比较常用的简单功能,并可以很容易的被重用或者是插入其他应用程序中去。所有遵循一定编程原则的Java类都可以被称作JavaBean。 一.JavaBean技术概述       JavaBean是基于Java的组件模型,由属性、方法和事件3部

    2022年7月8日
    24
  • pycharm 设置环境变量

    pycharm 设置环境变量python 使用 pycharmIDE 开发中 设置到环境的切换 我使用环境变量控制的方法 环境变量设置方法 Run gt Editconfigur 中点击 Environmentv 后的目录标志加入环境变量注意 加入环境变量后 要重启服务 不然不会读取到新的环境变量值

    2026年3月27日
    2
  • IE8下textarea的onpropertychange问题Stack overflow at line.「建议收藏」

    IE8下textarea的onpropertychange问题Stack overflow at line.「建议收藏」代码如下:      Document            body{background:#fff;}      textarea{width:300px;min-height:60px;overflow:hidden;resize:none;}            $(function(){

    2022年7月15日
    19
  • photoshop是目前图像处理理最顶级的一款软件_图片处理软件app

    photoshop是目前图像处理理最顶级的一款软件_图片处理软件app说到照片和图像编辑/操纵,真的没有更好的应用,AdobePS图象处理软件。 摄影师和创意工作室会同意这是总理的照片编辑应用期。不幸的是,PS图象处理软件还配备了一个陡峭的学习曲线和价格标签,我们必须考虑我们不会使用矫枉过正的1/3的功能包装成PS图象处理软件。 记住让我们看一些免费的在线和客户端安装的应用程序可以在互联网上看。 选项是巨大的所以我缩小了我的审查,只有最好的在线和客户端安装

    2022年4月20日
    61
  • 【python教程入门学习】PyCharm 如何使用

    【python教程入门学习】PyCharm 如何使用PyCharm 具体介绍 PyCharm 是由 JetBrains 打造的一款 PythonIDE 我们知道 VS2010 的重构插件 Resharper 就是出自 JetBrains 之手 那么 PyCharm 有什么吸引人的特点呢 首先 PyCharm 用于一般 IDE 具备的功能 比如 调试 语法高亮 Project 管理 代码跳转 智能提示 自动完成 单元测试 版本控制 另外 PyCharm 还提供了一些很好的功能用于 Django 开发 同时支持 GoogleAppEng 更酷的是 PyCharm 支持 IronPy

    2026年3月27日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号