当前位置: 首页 > article >正文

【摄影测量】从零实现张正友标定法:手写代码解析相机内参/外参与畸变校正

1. 从棋盘格到数学模型张正友标定法基础第一次接触相机标定时我被那些复杂的数学符号吓到了。直到自己动手实现了一遍张正友标定法才发现它的精妙之处其实非常直观。想象你手里拿着一个国际象棋棋盘用手机从不同角度拍摄它——这就是张正友标定法需要的全部材料。棋盘格的每个角点都是已知的世界坐标点而它们在照片中的位置可以通过OpenCV的findChessboardCorners()轻松获取。这两个信息之间藏着相机的全部秘密。我常用的棋盘格是9x6的网格内角点8x5打印在A4纸上就能用。关键是要保证棋盘格平整实测发现贴在玻璃板上效果最好。单应性矩阵(Homography)是这个过程中的关键桥梁。它建立了棋盘格平面Z0的世界坐标系与图像平面之间的映射关系。计算单应性矩阵时我习惯用DLT直接线性变换算法。下面是核心代码片段def compute_homography(world_points, image_points): A [] for (x, y), (u, v) in zip(world_points, image_points): A.append([x, y, 1, 0, 0, 0, -u*x, -u*y, -u]) A.append([0, 0, 0, x, y, 1, -v*x, -v*y, -v]) A np.array(A) _, _, V np.linalg.svd(A) H V[-1].reshape(3, 3) return H / H[2,2] # 归一化这个函数输入世界坐标和对应的图像坐标输出3x3的单应性矩阵。注意最后要进行归一化处理让H[2,2]1。在实际项目中我通常会加入RANSAC来剔除异常点提高鲁棒性。2. 内参约束与闭合解计算拿到单应性矩阵后真正的魔术开始了。张正友的聪明之处在于发现了内参矩阵的约束条件。相机内参矩阵K通常长这样[fx 0 cx] [0 fy cy] [0 0 1]其中fx,fy是焦距cx,cy是主点坐标。通过定义B(K⁻ᵀ)K⁻¹我们可以建立两个关键约束h₁ᵀBh₂ 0h₁ᵀBh₁ h₂ᵀBh₂每个单应性矩阵可以提供这两个方程而B有6个未知数因为对称。所以至少需要3张不同角度的棋盘格图片才能求解。我建议用10-15张图片这样结果更稳定。具体实现时我建了一个方程组求解器def solve_intrinsics(homographies): V [] for H in homographies: h1, h2 H[:,0], H[:,1] V.append([h1[0]*h2[0], h1[0]*h2[1]h1[1]*h2[0], h1[1]*h2[1], h1[2]*h2[0]h1[0]*h2[2], h1[2]*h2[1]h1[1]*h2[2], h1[2]*h2[2]]) V.append([(h1[0]**2-h2[0]**2), 2*(h1[0]*h1[1]-h2[0]*h2[1]), (h1[1]**2-h2[1]**2), 2*(h1[0]*h1[2]-h2[0]*h2[2]), 2*(h1[1]*h1[2]-h2[1]*h2[2]), (h1[2]**2-h2[2]**2)]) _, _, V np.linalg.svd(np.array(V)) b V[-1] B np.array([[b[0], b[1], b[3]], [b[1], b[2], b[4]], [b[3], b[4], b[5]]]) return B解出B后通过Cholesky分解就能得到内参矩阵K。这里有个坑要确保B是正定矩阵否则分解会失败。我在实际项目中遇到过这个问题最后是通过增加标定图片数量解决的。3. 外参提取与初始畸变估计有了内参矩阵K外参就很容易计算了。对于每个单应性矩阵H我们可以求出对应的旋转矩阵R和平移向量tdef compute_extrinsics(K, H): K_inv np.linalg.inv(K) h1, h2, h3 H[:,0], H[:,1], H[:,2] lambda_ 1 / np.linalg.norm(K_inv h1) r1 lambda_ * K_inv h1 r2 lambda_ * K_inv h2 r3 np.cross(r1, r2) t lambda_ * K_inv h3 R np.column_stack((r1, r2, r3)) # 通过SVD保证R是旋转矩阵 U, _, Vt np.linalg.svd(R) R U Vt return R, t这里有个关键点理论上r1和r2应该是正交的但由于噪声存在直接计算出的R可能不满足旋转矩阵的性质行列式1。我通过SVD分解进行了修正确保得到的R是真正的旋转矩阵。畸变参数初始估计也很重要。我通常先假设只有径向畸变(k1,k2)用线性最小二乘求解def estimate_distortion(pts_2d, pts_3d, K, dist_coeffs): # 将3D点投影到图像平面 rvecs [cv2.Rodrigues(R)[0] for R in rotations] tvecs translations obj_points [pts_3d] * len(rvecs) # 只优化k1,k2 flags cv2.CALIB_FIX_K3 | cv2.CALIB_FIX_TANGENT_DIST ret, _, _, _, dist cv2.calibrateCamera( obj_points, pts_2d, image_size, K, None, flagsflags, criteria(cv2.TERM_CRITERIA_EPS cv2.TERM_CRITERIA_MAX_ITER, 30, 1e-6)) return dist这个初始估计会作为后续非线性优化的起点。在实际项目中我发现如果初始畸变估计偏差太大后续优化可能陷入局部最优。4. 非线性优化与实战技巧前面的步骤得到的参数还比较粗糙需要用Levenberg-Marquardt算法进行非线性优化。优化目标是最小化重投影误差def reprojection_error(params, world_points, image_points, num_views): # 解包参数 fx, fy, cx, cy, k1, k2 params[:6] K np.array([[fx, 0, cx], [0, fy, cy], [0, 0, 1]]) dist np.array([k1, k2, 0, 0]) # 计算重投影误差 error 0 for i in range(num_views): rvec params[6i*6 : 9i*6] tvec params[9i*6 : 12i*6] proj_points, _ cv2.projectPoints(world_points, rvec, tvec, K, dist) error np.sum((image_points[i] - proj_points.squeeze())**2) return error这里我使用了scipy的least_squares优化器。实测发现将旋转向量表示为罗德里格斯向量而不是旋转矩阵可以减少参数数量提高优化效率。几个实战中总结的技巧标定板要尽量充满画面但所有角点必须清晰可见拍摄角度要多样化包含俯视、仰视、倾斜等多种姿势光照要均匀避免反光和阴影影响角点检测至少需要10-15张高质量图片20张以上效果更好重投影误差最好控制在0.1像素以下我曾经在一个工业检测项目中由于现场光线问题导致标定不稳定。后来改用红外棋盘格和红外相机问题迎刃而解。这提醒我们实际应用中要根据环境特点灵活调整方案。5. 结果验证与可视化标定完成后验证环节必不可少。我通常会做三件事检查重投影误差将标定板角点重新投影到图像上计算与检测角点的距离可视化畸变校正效果用cv2.undistort观察校正前后的图像变化测试外参稳定性移动标定板观察外参变化是否符合预期下面是一个简单的可视化代码def visualize_calibration(img, K, dist): h, w img.shape[:2] new_K, roi cv2.getOptimalNewCameraMatrix(K, dist, (w,h), 1, (w,h)) undistorted cv2.undistort(img, K, dist, None, new_K) plt.figure(figsize(12,6)) plt.subplot(121) plt.imshow(img) plt.title(原始图像) plt.subplot(122) plt.imshow(undistorted) plt.title(校正后图像) plt.show()在无人机视觉导航项目中我发现虽然标定时的重投影误差很小但实际使用时仍有明显偏差。后来发现是因为镜头对焦距离变化导致内参改变。于是改为在不同对焦距离下分别标定使用时根据对焦距离插值得到内参效果显著改善。标定结果的精度评估也很重要。除了重投影误差我还会计算参数的不确定性。通过雅可比矩阵的奇异值分解可以估计各个参数的置信区间。这在需要高精度的测量应用中尤为重要。

相关文章:

【摄影测量】从零实现张正友标定法:手写代码解析相机内参/外参与畸变校正

1. 从棋盘格到数学模型:张正友标定法基础 第一次接触相机标定时,我被那些复杂的数学符号吓到了。直到自己动手实现了一遍张正友标定法,才发现它的精妙之处其实非常直观。想象你手里拿着一个国际象棋棋盘,用手机从不同角度拍摄它—…...

从VGG到ResNet:我的模型为什么越深效果越差?深入对比两种经典网络的设计哲学与实战选择

从VGG到ResNet:深度神经网络的设计哲学与实战选择指南 当你第一次尝试用VGG16完成图像分类任务时,可能会惊讶于它的表现——直到你发现训练更深的VGG19时,准确率不升反降。这种反直觉的现象引出了深度学习领域的一个核心问题:为什…...

向量运算的几何奥秘:叉积与点积的混合运算规则解析

1. 从几何视角理解向量运算的本质 第一次接触向量运算时,很多人会被各种公式绕得头晕。其实换个角度看,这些运算规则都对应着直观的几何现象。就像小时候玩积木,看似简单的拼接背后藏着空间结构的奥秘。 点积像是测量两个向量的"重合度&…...

从音频到全身动捕:手把手教你用AudCast和DITs生成带手势的AI视频(附开源项目分析)

从音频到全身动捕:手把手教你用AudCast和DITs生成带手势的AI视频(附开源项目分析) 在数字内容创作领域,AI视频生成技术正经历从静态图像到动态交互的跨越式发展。传统音频驱动视频方案往往局限于面部表情同步,而全身动…...

Golang笔记1-变量与类型

Go 变量与类型 1. 怎么声明变量 // var 写法:可以在函数外用(全局) var name string "张三" var age int 25 var isAdmin bool // 不赋值就是零值// : 短声明:只能在函数内用(日常首选) name :…...

Ostrakon-VL-8B多模态运维监控实战:智能日志分析与故障预警

Ostrakon-VL-8B多模态运维监控实战:智能日志分析与故障预警 最近和几个做运维的朋友聊天,大家普遍都在吐槽一件事:每天上班就像在“看监控”和“查日志”之间来回切换。服务器告警一响,就得一头扎进海量的日志文件里,…...

深刻理解虚拟内存机制

注意:页框和页大小一样,只是为了区分物理和虚拟,本文统称为页 声明:本文借鉴参考小林coding和鸭大坑导进行整合,有些个人理解,站在巨人的肩膀上学习 文章目录为什么要有虚拟内存怎么解决上述问题&#xff1…...

抢救你的数字青春:QQ空间记忆永久保存全攻略

抢救你的数字青春:QQ空间记忆永久保存全攻略 【免费下载链接】GetQzonehistory 获取QQ空间发布的历史说说 项目地址: https://gitcode.com/GitHub_Trending/ge/GetQzonehistory 当你在整理旧物时偶然翻到泛黄的毕业照,是否会想起QQ空间里那些更鲜…...

构建一个抗揍的 Go TCP 聊天服务:异常兜底与防御性编程实践

构建一个抗揍的 Go TCP 聊天服务:异常兜底与防御性编程实践 在用 Go 实现一个简单的 TCP 聊天室时,实现“上线、下线、广播、私聊”等功能并不难。但如果要把它放到公网,面对真实网络环境中的网络抖动、恶意攻击(如超长消息洪水、…...

三步搞定空洞骑士模组管理:Scarab让复杂依赖关系变得简单

三步搞定空洞骑士模组管理:Scarab让复杂依赖关系变得简单 【免费下载链接】Scarab An installer for Hollow Knight mods written in Avalonia. 项目地址: https://gitcode.com/gh_mirrors/sc/Scarab 还在为《空洞骑士》模组安装的各种技术难题而头疼吗&…...

Qt+OpenGL实战:从SOLIDWORKS到UR3机械臂OBJ模型渲染全流程

QtOpenGL实战:从SOLIDWORKS到UR3机械臂OBJ模型渲染全流程 在机器人仿真开发领域,将工业设计软件中的精密模型转化为可交互的三维可视化应用是一个关键且具有挑战性的环节。UR3协作机械臂作为工业自动化领域的明星产品,其高精度模型的渲染与操…...

开源抽卡模拟器:浏览器中的原神资源策略实验室

开源抽卡模拟器:浏览器中的原神资源策略实验室 【免费下载链接】Genshin-Impact-Wish-Simulator Best Genshin Impact Wish Simulator Website, no need to download, 100% running on browser! 项目地址: https://gitcode.com/gh_mirrors/gen/Genshin-Impact-Wis…...

Java SpringBoot+Vue3+MyBatis 图书进销存管理系统系统源码|前后端分离+MySQL数据库

摘要 随着信息技术的快速发展,传统图书进销存管理方式逐渐暴露出效率低下、数据冗余和人工操作繁琐等问题。图书行业对高效、精准的管理系统需求日益增长,尤其在库存管理、销售统计和数据分析方面,亟需一套智能化解决方案。基于前后端分离架构…...

PPOCRLabel快捷键全解析:告别鼠标点点点,提升标注效率的隐藏技巧

PPOCRLabel快捷键全解析:告别鼠标点点点,提升标注效率的隐藏技巧 当你面对上千张待标注的图片时,每次点击菜单、切换工具、调整选框的微小延迟,都会累积成惊人的时间损耗。专业标注员的秘密武器从来不是鼠标,而是那些藏…...

keil工程创建常见问题

问题描述 keil工程文件创建遇到十八个错误: 例如:./Start/core_cm3.h(1756): error: expected ‘;’ after top level declarator static __INLINE uint32_t ITM_SendChar (uint32_t ch)解决方案:提示:点击魔术棒→Target→Code G…...

广西大学电气专业课设资料包|短路计算课程设计全套(含源码+实验报告+理论PPT)

温馨提示:文末有联系方式广西大学电气专业课程设计资料合集 专注服务广大学生,精心整理广西大学电气工程及其自动化专业核心课设,覆盖课程设计全流程需求。短路电流计算课程设计全套电子资料 包含完整可编译运行的软件程序(支持主…...

【VBA】【EXCEL】分类汇总

option explicit option base 1Sub 分类汇总()Dim ws0 As Worksheet, ws1 As WorksheetDim arr0 As Variant, arr1 As VariantDim lastRow As Long, i As Long, m As Long, cnt As LongDim acct As String, opp As String, key As String, pts() As StringDim amt As Double, t…...

内容管理系统 CMS 发展史:从静态建站到 2026 智能一体化协同平台

内容管理系统(CMS)作为支撑互联网内容生态的核心基础软件,自诞生以来已走过 30 余年历程。它始终紧跟技术浪潮与市场需求,从最初简单的静态页面制作工具,逐步演进为集内容管理、低代码开发、智能分析、多端分发于一体的…...

2025届最火的六大降重复率助手推荐

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 能降低AIGC检测率的关键之处在于模拟人类写作所具备的自然性以及逻辑跳跃。其一,…...

2026届学术党必备的十大降重复率神器解析与推荐

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 减少AIGC率的关键要点是全力去降低文本里那些能够被分辨出来的机器生成特性 ,这…...

2025届最火的降AI率神器推荐榜单

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 近期,知网发布了有关人工智能生成内容,也就是AIGC的检测服务以及使用…...

UART 入门指南(Linux新手版)

UART 入门指南(Linux新手版) 适用对象:嵌入式/电子/通信初学者 目录 什么是 UARTUART 工作原理硬件接口与接线通信参数详解编程示例常见问题与调试 1. 什么是UART 1.1 基本定义 UART 的全称是 Universal Asynchronous Receiver/Transmitte…...

C++20 协同调度原语:利用 std::atomic::wait/notify 实现低功耗自旋锁在高并发下的快速响应协议

各位同仁,女士们,先生们,欢迎来到今天的技术讲座。在现代C编程中,高性能与低功耗的追求从未停止。随着多核处理器的普及和异步编程模型的兴起,对并发原语的精细化控制变得尤为关键。C20标准为我们带来了诸多激动人心的…...

C++ 硬件特征自适应分发:利用 C++ 特性实现对不同 CPU 指令集(AVX2/AVX-512)的运行时代码路径最优选择

C 硬件特征自适应分发:运行时代码路径最优选择各位技术爱好者,大家好!在现代高性能计算领域,充分挖掘硬件潜力是提升程序性能的关键。我们知道,CPU架构在不断演进,其指令集也在持续扩展,以支持更…...

uniapp实战:uview Collapse组件动态数据加载后高度异常的3种解决方案

Uniapp实战:uView Collapse组件动态数据加载后高度异常的深度解决方案 在Uniapp开发中,uView UI库的Collapse折叠面板组件因其简洁易用而广受欢迎。但当我们需要动态加载数据并展开面板时,经常会遇到一个棘手的问题:面板高度计算不…...

ROS2 Jazzy机器人导航避坑指南:详解Navigation2参数配置中那些容易出错的‘坑’

ROS2 Jazzy导航系统参数配置实战:从踩坑到精通的避坑手册 当你第一次打开ROS2 Jazzy的Navigation2参数配置文件时,是否感觉像是面对一本没有注释的古老秘籍?那些看似简单的参数背后,往往隐藏着让机器人"发疯"的陷阱。本…...

在PhpStudy中进行PHP版本切换的详细流程(Linux和Windows)

在使用多样化的 PHP Web 应用程序时,选择合适的 PHP 版本至关重要。例如,一些老旧的应用程序可能是基于早期版本的 PHP 开发的,如果使用最新版本的 PHP 来运行,可能会遇到兼容性问题,导致错误。反之,如果用…...

PHP中比较两个对象的几种方式小结

在PHP中,比较两个对象并不是一件直接明了的事情,因为对象之间的比较通常依赖于它们的属性和状态,而这些属性和状态可能非常复杂且多样化。PHP提供了几种方式来比较对象,但每种方式都有其特定的用途和限制。1. 使用和运算符在PHP中…...

PHP脚本设置无限执行时间的四种方法

为 PHP 脚本设置无限执行时间是一个在特定场景下可能需要的操作,比如执行长时间运行的后台任务、数据迁移、大批量数据处理等。然而,值得注意的是,设置无限执行时间并不是一种推荐的做法,因为它可能导致服务器资源被长时间占用&am…...

ThinkPHP实现定时任务的操作步骤

到一个需求:定时检查设备信息,2分钟没有心跳的机器,推送消息给相关人员,用thinkphp5框架,利用框架自带的任务功能与crontab配合来完成定时任务。第一步:分析需求先写获取设备信息,2分钟之内没有…...