opencv 实现两个图片的拼接去重功能

基础知识介绍

cv::Mat 是OpenCV库中用来表示图像和矩阵数据的核心类之一。它是一个多维数组,可以存储图像像素数据、矩阵数据以及其他类型的数据。以下是关于 cv::Mat 类的一些详细解释:

  1. 构造函数cv::Mat 类有多个构造函数,可以用来创建不同类型的矩阵。常用的构造函数包括:

    • cv::Mat(): 默认构造函数,创建一个空的矩阵。
    • cv::Mat(int rows, int cols, int type): 创建指定行数、列数和数据类型的矩阵。
    • cv::Mat(cv::Size size, int type): 根据指定的大小和数据类型创建矩阵。
    • cv::Mat(int rows, int cols, int type, void* data, size_t step): 使用指定的数据指针和步长创建矩阵。
  2. 成员函数cv::Mat 类提供了许多成员函数,用于对矩阵进行操作,包括:

    • rows(): 返回矩阵的行数。
    • cols(): 返回矩阵的列数。
    • size(): 返回矩阵的大小。
    • at<>(): 访问矩阵中的像素值。
    • clone(): 复制矩阵。
    • copyTo(): 将矩阵复制到另一个矩阵。
    • convertTo(): 转换矩阵的数据类型。
  3. 数据访问:可以通过 at<>() 函数或者使用 Mat::ptr 来访问矩阵中的像素值。例如,mat.at<uchar>(i, j) 可以获取矩阵中第 i 行、第 j 列的像素值。

  4. 数据类型cv::Mat 支持多种数据类型,包括整数、浮点数等。数据类型由 type() 函数返回,常见的数据类型包括 CV_8UCV_32F 等。

  5. 图像处理cv::Mat 类经常用于图像处理,可以通过 imread() 读取图像文件,通过 imwrite() 写入图像文件,还可以进行图像的剪裁、缩放、旋转等操作。

cv::Rect 是OpenCV库中用来表示矩形区域的类。它通常用于定义矩形的位置和大小,以便在图像处理中进行裁剪、绘制等操作。以下是关于 cv::Rect 类的一些详细解释:

  1. 构造函数cv::Rect 类有多个构造函数,可以用来创建不同类型的矩形。常用的构造函数包括:

    • cv::Rect(): 默认构造函数,创建一个空的矩形。
    • cv::Rect(int x, int y, int width, int height): 创建一个指定位置和大小的矩形。
    • cv::Rect(cv::Point pt1, cv::Point pt2): 根据两个点的坐标创建矩形,其中一个点表示左上角,另一个点表示右下角。
  2. 成员变量cv::Rect 类包含以下成员变量:

    • x: 矩形左上角的 x 坐标。
    • y: 矩形左上角的 y 坐标。
    • width: 矩形的宽度。
    • height: 矩形的高度。
  3. 成员函数cv::Rect 类提供了一些成员函数,用于方便地操作矩形,包括:

    • area(): 返回矩形的面积。
    • contains(): 判断一个点是否在矩形内部。
    • tl(): 返回矩形的左上角点。
    • br(): 返回矩形的右下角点。
    • size(): 返回矩形的大小。
  4. 使用cv::Rect 类通常用于定义感兴趣区域(Region of Interest,ROI),可以通过矩形来指定图像中的一个区域,然后进行裁剪、绘制等操作。

两个图片的拼接

int main()
{//输入两个图片cv::Mat image1 = cv::imread("../../picture/1.jpg");cv::Mat image2 = cv::imread("../../picture/2.jpg");//新图片的长宽int newWidth = image1.cols + image2.cols;int newHeight = std::max(image1.rows, image2.rows);//初始化结果图像cv::Mat resultImage(newHeight, newWidth, image1.type());//复制图像内容cv::Mat left(resultImage, cv::Rect(0, 0, image1.cols, image1.rows));image1.copyTo(left);cv::Mat right(resultImage, cv::Rect(image1.cols, 0, image2.cols, image2.rows));image2.copyTo(right);//保存图像cv::imwrite("../../picture/result.jpg", resultImage);

图片去重

int quchong()
{cv::Mat image1 = cv::imread("../../picture/1.jpg");cv::Mat image2 = cv::imread("../../picture/2.jpg");//夜色空间转换cv::Mat grayImage1, grayImage2;cv::cvtColor(image1, grayImage1, cv::COLOR_BGR2GRAY);cv::cvtColor(image2, grayImage2, cv::COLOR_BGR2GRAY);cv::imwrite("../../picture/BGR2GRAY.jpg", grayImage1);// 使用ORB特征检测器和描述符进行特征提取cv::Ptr<cv::ORB> orb = cv::ORB::create();std::vector<cv::KeyPoint> keypoints1, keypoints2;cv::Mat descriptors1, descriptors2;orb->detectAndCompute(image1, cv::noArray(), keypoints1, descriptors1);orb->detectAndCompute(image2, cv::noArray(), keypoints2, descriptors2);cv::imwrite("../../picture/descriptors1.jpg", descriptors1);// 使用暴力匹配器进行特征匹配cv::BFMatcher matcher(cv::NORM_HAMMING);std::vector<cv::DMatch> matches;matcher.match(descriptors1, descriptors2, matches);// 筛选匹配点std::vector<cv::Point2f> matchedPoints1, matchedPoints2;for (const cv::DMatch& match : matches) {matchedPoints1.push_back(keypoints1[match.queryIdx].pt);matchedPoints2.push_back(keypoints2[match.trainIdx].pt);}// 计算透视变换矩阵cv::Mat H = cv::findHomography(matchedPoints2, matchedPoints1, cv::RANSAC);cv::imwrite("../../picture/H.jpg", H);// 对第二张图像进行透视变换cv::Mat warpedImage2;cv::warpPerspective(image2, warpedImage2, H, cv::Size(image1.cols + image2.cols, image1.rows));cv::imwrite("../../picture/warpedImage2.jpg", warpedImage2);// 将两张图像拼接在一起cv::Mat resultImage = warpedImage2.clone();image1.copyTo(resultImage(cv::Rect(0, 0, image1.cols, image1.rows)));cv::imwrite("../../picture/result.jpg", resultImage);return 0;
}

实验结果

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Linux】常用的命令

文章目录 lsls -l / touchcdpwdcatechovim打开文件编辑内容保存退出 mkdirrmmvcpmangreppsnetstat总结 &#xff1a; ls ls > list 列出当前目录下都有哪些内容&#xff08;文件/目录&#xff09; 直接输入 ls&#xff0c;是查看当前目录的情况&#xff1b;输入 ls/ 就是看…

数据结构 --- 二叉树

一、满二叉树 在一棵二叉树中&#xff0c;如果所有分支节点都存在左子树和右子树&#xff0c;并且所有叶子节点都在同一层上&#xff0c;这 样的二叉树称为满二叉树。 每层节点数量为 2 ^ (n - 1) &#xff08;n为层数&#xff09; 总节点个数为 2 ^ n - 1 二、完全…

【Java】基于JWT+Token实现完整登入功能(原理+实操图解)

Java系列文章目录 补充内容 Windows通过SSH连接Linux 第一章 Linux基本命令的学习与Linux历史 文章目录 Java系列文章目录一、前言二、学习内容&#xff1a;三、问题描述四、解决方案&#xff1a;4.1 认识相关依赖4.1.1 工具包依赖4.1.2 非空注解依赖4.1.3 Token相关依赖4.1.4…

【正式版】深度技术Win10系统22H2最新版本:免费下载!

今日&#xff0c;系统之家小编给大家分享2024年最新发布的深度技术Win10正式版系统&#xff0c;该版本系统基于微软官方最新Windows10 22H2 19045.4842 64位专业版进行离线制作&#xff0c;确保安全无病毒&#xff0c;且修复了部分系统Bug&#xff0c;整体操作体验感更出色。系…

6.1排序——插入排序与希尔排序

本篇博客来梳理两种常见排序算法&#xff1a;插入排序与希尔排序 常见的排序算法如图 写排序算法的原则&#xff1a;先写单趟&#xff0c;再写整体 一、直接插入排序 1&#xff0e;算法思想 先假定第一个数据有序&#xff0c;把第二个数据插入&#xff1b;再假设前两个数据…

[【人工智能学习笔记】4_3 深度学习基础之卷积神经网络

卷积神经网络概述 卷积神经网络(Convolutional Neural Network, CNN)一种带有卷积结构的深度神经网络,通过特征提取和分类识别完成对输入数据的判别;在1989年提出,早期被成功用于手写字符图像识别;2012年更深层次的AlexNet网络取得成功,伺候卷积神经网络被广泛应用于各…

uniapp使用uni-popup做底部弹出选项(vue3)

效果图 页面代码 <!-- 发票筛选弹出框 --><uni-popup ref"popupRef" type"bottom" border-radius"10px 10px 0 0" background-color"#fff"><h4 style"text-align: center;margin-bottom: 20px;">发票筛…

node解析dxf文件

1、dxf数据说明 DXF是一种开放的矢量数据格式&#xff0c;可以分为两类&#xff1a;ASCII格式和二进制格式&#xff1b;ASCII具有可读性好的特点&#xff0c;但占用的空间较大&#xff1b;二进制格式则占用的空间小、读取速度快。由于AutoCAD是最流行的CAD系统&#xff0c;DXF也…

uniapp 懒加载、预加载、缓存机制深度解析

uniapp 懒加载、预加载、缓存机制深度解析 文章目录 uniapp 懒加载、预加载、缓存机制深度解析一、为什么要使用uniapp的懒加载、预加载和缓存机制二、如何使用uniapp的懒加载、预加载和缓存机制1. 懒加载2. 预加载3. 缓存机制 四、扩展与高级技巧1. 结合懒加载和预加载优化页面…

眼科市场格局固化,排名靠后的光正眼科还能逆袭吗?

眼科是A股的热门领域&#xff0c;也是医疗的黄金赛道。或许也正因为如此&#xff0c;这条赛道已经习惯了通过并购&#xff0c;利用资本杠杆跑马圈地。以最大规模的龙头爱尔眼科为首&#xff0c;并购是眼科的常规操作。 然而&#xff0c;真正观察赛道腰部及以下的公司&#xff…

elementUI根据列表id进行列合并@莫成尘

本文章提供了elementUI根据列表id进行列合并的demo&#xff0c;效果如图&#xff08;可直接复制代码粘贴&#xff09; <template><div id"app"><el-table border :data"tableList" style"width: 100%" :span-method"objectS…

2024.9.9营养小题【2】

营养&#xff1a; 1、什么数是丑数&#xff1f; 2、数学数学&#xff0c;丑数的数学意义&#xff0c;哎&#xff0c;数学思维我是忘干净了。 3、可以把while循环换成for循环。由此又想到了一点&#xff0c;三个循环结构各有使用场景。 for(;n%factors[i]0;n/factors[i]){}

网络编程day02(字节序、TCP编程)

目录 【1】字节序 1》大小端转换 2》端口转换 3》IP地址转换 主机字节序转换为网络字节序 &#xff08;小端序->大端序&#xff09; 网络字节序转换为主机字节序&#xff08;大端序->小端序&#xff09; 【2】TCP编程 1》流程 2》函数接口 1> socket 2> …

C# 删除Word文档中的段落

在编辑Word文档时&#xff0c;我们有时需要调整段落的布局、删除不必要的段落以优化文档的结构和阅读体验。本文将通过以下3个简单示例演示如何使用免费.NET库删除Word文档中的段落 。 目录 C# 删除Word中的指定段落 C# 删除Word中的所有段落 C# 删除Word中的空白段落 免费…

分组注解和自定义注解及分页查询

自定义注解的使用步骤 案例&#xff1a; 此时state需要进行的校验使用普通方式无法满足&#xff0c;需要我们根据需求进行自定义注解 创建一个注解 Documented//元注解 Retention(RetentionPolicy.RUNTIME)//元注解 Constraint(validatedBy {StateValidation.class}//指定提供…

网络编程day04(UDP、Linux IO 模型)

目录 【1】UDP 1》通信流程 2》函数接口 1> recvfrom 2> sendto 3》代码展示 1> 服务器代码 2> 客户端代码 【2】Linux IO 模型 场景假设一 1》阻塞式IO&#xff1a;最常见、效率低、不耗费CPU 2》 非阻塞 IO&#xff1a;轮询、耗费CPU&#xff0c;可以处…

Java后台生成二维码

一、效果图 二、实现代码 1.添加依赖 <!-- zxing生成二维码 --> <dependency><groupId>com.google.zxing</groupId><artifactId>core</artifactId><version>3.3.3</version> </dependency><dependency><grou…

大数据之Flink(四)

11、水位线 11.1、水位线概念 一般实时流处理场景中&#xff0c;事件时间基本与处理时间保持同步&#xff0c;可能会略微延迟。 flink中用来衡量事件时间进展的标记就是水位线&#xff08;WaterMark&#xff09;。水位线可以看作一条特殊的数据记录&#xff0c;它是插入到数…

机械学习—零基础学习日志(Python做数据分析02)

现在开始使用Python尝试做数据分析。具体参考的网址链接放在了文章末尾。 引言 我通过学习《利用Python进行数据分析》这本书来尝试使用Python做数据分析。书里让下载&#xff0c;anaconda&#xff0c;使用Jupyter来写代码&#xff0c;只是下载一个anaconda的确有点费时间&am…

计算机的发展史和基本结构

好久不见&#xff0c;粉粉们&#xff0c;我是#Y清墨。今天来分享一下最近学习做的笔记。 计算机发展史和四代计算机概述 阶段 年代 电子元件 运算速度&#xff08;每秒/次&#xff09; 第一代 1946-1958 真空电子管 数千至数万 第二代 1958-1964 晶体管 几十万至百万…