【opencv】教程代码 —features2D(6)透视矫正:读取两个棋盘格图片并进行图像对齐...

news2024/12/24 9:25:12

perspective_correction.cpp 透视校正

c0142c5f74bb5c10afcd4605746fea53.png

hconcat(img2, img1_warp, img_draw_warp);

51b7f1ab0528d54fb8e70eded1e47285.png

hconcat(img1, img2, img_draw_matches);

20469dffb54bcda2a5fbd5004e8a4712.png

#include <iostream> // 引入iostream库,用于进行标准输入和输出操作
#include <opencv2/core.hpp> // 引入opencv的core库,含有OpenCV的基础结构和操作
#include <opencv2/imgproc.hpp> // 引入opencv的imgproc库,包含了图像处理的各种操作
#include <opencv2/calib3d.hpp> // 引入opencv的calib3d库,包含3D校准相关的功能
#include <opencv2/highgui.hpp> // 引入opencv的highgui库,提供了显示和读写功能


using namespace std; // 使用std标准库的名字空间
using namespace cv; // 使用OpenCV库的名字空间


namespace // 创建匿名名字空间
{
enum Pattern { CHESSBOARD, CIRCLES_GRID, ASYMMETRIC_CIRCLES_GRID }; // 定义Pattern枚举,包含三种模式:CHESSBOARD(棋盘格)、CIRCLES_GRID(圆形网格)、ASYMMETRIC_CIRCLES_GRID(不对称圆形网格)


Scalar randomColor( RNG& rng ) // 函数定义,生成随机颜色 
{
  int icolor = (unsigned int) rng; // 将RNG对象转为unsigned int类型的值
  return Scalar( icolor & 255, (icolor >> 8) & 255, (icolor >> 16) & 255 ); // 创建并返回一个颜色Scalar对象(B、G、R格式)
}


void perspectiveCorrection(const string &img1Path, const string &img2Path, const Size &patternSize, RNG &rng) // 定义透视修正函数,输入参数为两张图片路径、模式尺寸及RNG对象
{
    Mat img1 = imread( samples::findFile(img1Path) ); // 读取图像1
    Mat img2 = imread( samples::findFile(img2Path) ); // 读取图像2


    // 查找棋盘格角点
    vector<Point2f> corners1, corners2; // 创建两个浮点型点向量,存放两张图片的角点
    bool found1 = findChessboardCorners(img1, patternSize, corners1); // 在图像1中查找棋盘格的角点
    bool found2 = findChessboardCorners(img2, patternSize, corners2); // 在图像2中查找棋盘格的角点


    if (!found1 || !found2) // 如果在任何一张图片中都找不到角点
    {
        cout << "Error, cannot find the chessboard corners in both images." << endl; // 输出错误信息
        return; // 返回,结束程序
    }


    Mat H = findHomography(corners1, corners2); // 计算两组点之间的单应性矩阵H
    cout << "H:\n" << H << endl; // 打印出单应性矩阵H


    Mat img1_warp; // 创建一个新的Mat对象,用于存放透视变换后的图像
    warpPerspective(img1, img1_warp, H, img1.size()); // 对图像1进行透视变换


    // 拼接原图和变换后的图像
    Mat img_draw_warp; 
    hconcat(img2, img1_warp, img_draw_warp);
    imshow("Desired chessboard view / Warped source chessboard view", img_draw_warp); // 显示拼接后的图像


    // 绘制匹配的角点之间的线
    Mat img_draw_matches;
    hconcat(img1, img2, img_draw_matches); // 拼接两张原始图像
    for (size_t i = 0; i < corners1.size(); i++) // 遍历角点向量
    {
        Mat pt1 = (Mat_<double>(3,1) << corners1[i].x, corners1[i].y, 1); // 创建一个3x1的双精度浮点类型的矩阵,表示齐次坐标下的点
        Mat pt2 = H * pt1; // 使用单应性矩阵H对点进行转换
        pt2 /= pt2.at<double>(2); // 将转换后的点的坐标进行归一化
         // 计算映射后点的坐标位置
        Point end( (int) (img1.cols + pt2.at<double>(0)), (int) pt2.at<double>(1) );
        // 在图像上绘制代表匹配关系的线
        line(img_draw_matches, corners1[i], end, randomColor(rng), 2); 
    }


    imshow("Draw matches", img_draw_matches); // 显示匹配关系
    waitKey(); // 暂停程序,等待用户按键
}


const char* params
    = "{ help h         |       | print usage }"
      "{ image1         | left02.jpg | path to the source chessboard image }"
      "{ image2         | left01.jpg | path to the desired chessboard image }"
      "{ width bw       | 9     | chessboard width }"
      "{ height bh      | 6     | chessboard height }"; // 定义参数列表,包括帮助、图像路径、棋盘宽度和高度


int main(int argc, char *argv[]) // 主函数
{
    cv::RNG rng( 0xFFFFFFFF ); // 创建RNG对象
    CommandLineParser parser(argc, argv, params); // 创建命令行解析器


    if (parser.has("help")) // 如果用户输入了帮助参数
    {
        parser.about("Code for homography tutorial.\n"
            "Example 2: perspective correction.\n"); // 打印帮助信息
        parser.printMessage(); // 打印所有参数的信息
        return 0; // 程序正常结束
    }


    Size patternSize(parser.get<int>("width"), parser.get<int>("height")); // 从解析器获取棋盘格的宽度和高度,并构造Size对象
    perspectiveCorrection(parser.get<String>("image1"),
                          parser.get<String>("image2"),
                          patternSize, rng); // 执行透视校正操作


    return 0; // 程序正常结束
}

该段代码实现了对基于棋盘格模板的图片进行透视校正的功能。在实际操作中,先分别读取两幅图片,并在图片中找到棋盘角点,然后计算两组点对应的单应性矩阵(Homography),通过该矩阵实现透视变换,使其中一副图片的棋盘格对齐到另一幅图片中对应的位置,最后将变换效果进行显示。其中,代码中将原图与透视校正后的图像进行拼接显示,并绘制了角点匹配的连线,可以辅助观察效果。该代码通常应用于相机标定等领域。

H:
[0.3290339333220099, -1.244138808862929, 536.4769088231476;
 0.6969763913334047, -0.0893590907257152, -80.3406850408241;
 0.0004051172959296097, -0.001079740100565012, 1]

randomColor

783016cd91088d0bc09e66d9f70835fc.png

02a54095557e7ef8de6a477c73f11082.png

Mat H = findHomography(corners1, corners2);

831764504565f95ed04ad1892d644d42.png

Mat pt1 = (Mat_<double>(3,1) << corners1[i].x, corners1[i].y, 1);

11f6111e67375008be4ec94f9e44ece6.png

line(img_draw_matches, corners1[i], end, randomColor(rng), 2);

3787f9b43f93564c72cd63f3f561d184.png

48645d6675ed84201e760530d14f09c8.png

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

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

相关文章

session学习

3次请求均有sessionID session的作用 跟踪用户的行为&#xff0c;方便日后推荐客户端和服务器交互相对安全些session是代表会话&#xff0c;也可理解为客户端和服务端的交互sessionID是服务器生成的唯一字符串&#xff0c;用来跟踪用户行为cookie是浏览器自带的&#xff0c;专…

4.1作业

对菱形继承给出的代码中每一个类&#xff0c;写一个有参构造函数写出下列类的&#xff0c;构造函数(有参、无参)&#xff0c;析构函数&#xff0c;拷贝构造函数和拷贝赋值函数 class Father { int *p; const string name; } class Son:public Father { int *age; 3整理思维导图…

【JavaWeb】Day30.SpringBootWeb请求响应——响应

响应 HTTL协议的交互方式&#xff1a;请求响应模式&#xff08;有请求就有响应&#xff09;那么Controller程序&#xff0c;除了接收请求外&#xff0c;还可以进行响应。 1.ResponseBody 在我们前面所编写的controller方法中&#xff0c;都已经设置了响应数据。 controller方…

前端订阅推送WebSocket定时任务

0.需求 后端定时向前端看板推送数据&#xff0c;每10秒或者30秒推送一次。 1.前言知识 HTTP协议是一个应用层协议&#xff0c;它的特点是无状态、无连接和单向的。在HTTP协议中&#xff0c;客户端发起请求&#xff0c;服务器则对请求进行响应。这种请求-响应的模式意味着服务器…

路径规划——搜索算法详解(六):LPA*算法详解与Matlab代码

上文讲解了D*算法&#xff0c;D*算法为在动态环境下进行路径规划的场景提出了可行的解决方案&#xff0c;本文将继续介绍另外一种动态规划路径的方法——Lifelong Planning A*&#xff08;LPA*&#xff09;算法。 该算法可以看作是A*的增量版本&#xff0c;是一种在固定起始点…

语音克隆技术浪潮:探索OpenAI Voice Engine的奇妙之旅

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

Bridge Champ与Ignis公链:探索Web3游戏的新未来

在数字化和去中心化的浪潮中&#xff0c;Web3游戏与公链的融合为游戏行业带来了新的变革。特别是&#xff0c;Bridge Champ和Ignis公链的结合&#xff0c;展示了一种全新的游戏生态模式&#xff0c;不仅为玩家提供了更加公平、透明的游戏体验&#xff0c;同时也为游戏开发和运营…

Higress 基于自定义插件访问 Redis

作者&#xff1a;钰诚 简介 基于 wasm 机制&#xff0c;Higress 提供了优秀的可扩展性&#xff0c;用户可以基于 Go/C/Rust 编写 wasm 插件&#xff0c;自定义请求处理逻辑&#xff0c;满足用户的个性化需求&#xff0c;目前插件已经支持 redis 调用&#xff0c;使得用户能够…

C++ 注册Nacos

下载源码&#xff1a; git clone GitHub - nacos-group/nacos-sdk-cpp: C client for Nacos 编译源码 cd nacos-sdk-cpp cmake . make 生成库文件 在nacos-sdk-cpp 下 注册nacos 将include 和libnacos-cli.so libnacos-cli-static.a 放入你的工程 如果Nacos服务地址:…

3.26号arm

1. SPI相关理论 1.1 概述 spi是一种同步全双工串行总线&#xff0c;全称串行外围设备接口 通常SPI通过4个引脚与外部器件相连&#xff1a; MISO&#xff1a;主设备输入/从设备输出引脚。该引脚在从模式下发送数据&#xff0c;在主模式下接收数据。 MOSI&#xff1a;主设备输…

LangChain入门:9.使用FewShotPromptTemplate实现智能提示工程

在构建智能提示工程时&#xff0c;LangChain 提供了强大的 FewShotPromptTemplate 模型&#xff0c;它可以帮助我们更好地利用示例来指导大模型生成更加优质的提示。 在这篇博文中&#xff0c;我们将使用 LangChain 的 FewShotPromptTemplate 模型来设计一个智能提示工程&#…

StarRocks使用Minio备份和还原

1.安装minio minio api端口&#xff1a;9090 下文用到这个端口 必须提前创建好桶: packfdv5 名称自定义和后面对上就可以 2.创建备份仓库 格式&#xff1a; CREATE REPOSITORY <repository_name> WITH BROKER ON LOCATION "s3a://<bucket_name>/backup…

47.goto语句

目录 一.goto语句 二.语法格式 三.举例 四.视频教程 一.goto语句 goto语句可以使程序在没有任何条件的情况下跳转到指定位置&#xff0c;所以goto语句也就跳转语句。 二.语法格式 格式1&#xff1a;goto label&#xff1a;//其他代码 lable&#xff1a;//其他代码格式2&a…

【Node.js从基础到高级运用】二十、Node.js 强大的REPL

引言 Node.js REPL&#xff08;Read-Eval-Print Loop&#xff09;是一种交互式的命令行工具&#xff0c;它允许开发者快速地执行JavaScript代码&#xff0c;并查看结果。这个功能在进行快速原型设计、调试、学习JavaScript或Node.js时非常有用。 启动REPL 首先&#xff0c;确保…

【总结】在嵌入式设备上可以离线运行的LLM--Llama

文章目录 Llama 简介运用另一种&#xff1a;MLC-LLM 一个令人沮丧的结论在资源受限的嵌入式设备上无法运行LLM&#xff08;大语言模型&#xff09;。 一丝曙光&#xff1a;tinyLlama-1.1b&#xff08;10亿参数&#xff0c;需要至少2.98GB的RAM&#xff09; Llama 简介 LLaMA…

智慧安防监控EasyCVR视频调阅和设备录像回看无法自动播放的原因排查与解决

智慧安防监控EasyCVR视频管理平台能在复杂的网络环境中&#xff0c;将前端设备统一集中接入与汇聚管理。国标GB28181协议视频监控/视频汇聚EasyCVR平台可以提供实时远程视频监控、视频录像、录像回放与存储、告警、语音对讲、云台控制、平台级联、磁盘阵列存储、视频集中存储、…

Spring Boot项目启动速度优化

1、配置自动配置排除列表&#xff0c;减少启动自动配置扫描&#xff0c;配置项spring.autoconfigure.exclude 2、启动类添加索引注解Indexed&#xff0c;去除启动过程中 Components 的扫描步骤&#xff0c;直接从索引文件读取。 import org.springframework.stereotype.lndexe…

ansible-tower安装

特别注意&#xff1a;不需要提前安装ansible&#xff0c;因为ansible tower中的setup.sh脚本会下载对应的ansible版本 ansible tower不支持Ubuntu系统,对cenos系统版本也有一定的限制&#xff0c;建议使用centos7.9。 准备一台全新的机器安装&#xff0c;因为ansible tower需要…

第21章-直连路由和静态路由

1. 直连路由 1&#xff09;定义&#xff1a;指路由器接口直接相连的网段的路由&#xff1b; 2&#xff09;特点&#xff1a; ① 不需要特别的配置&#xff0c;双UP(物理层数据链路层)&#xff1b; ② 在路由器的接口上配置IP地址即可&#xff1b; ③ 开机自动产生&#xff1b; …

Docker容器赋能TitanIDE:引领编程新纪元的集成开发环境

Docker是一种容器化技术&#xff0c;它可以将应用程序和其所有的依赖项打包到一个轻量级、可移植的容器中。以下是Docker的基本概念和优势&#xff1a; 基本概念&#xff1a; 镜像&#xff08;Image&#xff09;&#xff1a;一个镜像是一个只读的模板&#xff0c;可以用于创建…