【OpenCV】广告屏换图 文稿矫正 透视变换 C++ 案例实现

2023-11-23 18:50

本文主要是介绍【OpenCV】广告屏换图 文稿矫正 透视变换 C++ 案例实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

前言

一、透视变换的概念

二、透视变换工作原理

三、相关函数

1.findHomography函数

2.warpPerspective函数

3.getPerspectiveTransform函数

四、透视变换案例

案例1:文稿矫正

案例2:广告屏换图

五、总结


前言

本文以实现广告屏换图为目标,学习OpenCV透视变化相关知识,核心的内容包括:OpenCV透视变化概念透视变化工作原理,以及广告屏换图文稿矫正实现 

一、透视变换的概念

  • 仿射变换透视变换在图像还原、图像局部变化处理方面有重要意义
  • 通常情况下,在2D平面中,仿射变换的应用较多,在3D平面中,透视变换又有了自己的一席之地,两种变换原理相似,结果也类似,可针对不同的场合使用适当的变换
  • 仿射变换透视变换的数学原理不需深究,在应用层面,仿射变换是图像基于3个固定顶点的变换,如下图所示:

二、透视变换工作原理

  • 透视变换:本质是将图像投影到一个新的视平面
  • 其通用变换公式,如下图所示:

  • (u,v)为原始图像像素坐标,(x=x’/w’,y=y’/w’)为变换之后的图像像素坐标
  • 相关公式,本文不做过多阐述,图解如下:

  • 仿射变换:可以理解为透视变换的特殊形式
  • 数学表达式,如下图所示:

 

  • 所以,给定透视变换对应的四对像素点坐标,即可求得透视变换矩阵
  • 反之,给定透视变换矩阵,即可对图像或像素点坐标完成透视变换
  • 如下图所示:

 

三、相关函数

1.findHomography函数

📌函数功能:找到两个平面之间的变换矩阵

📌参数:如下表所示

2.warpPerspective函数

📌函数功能:对图像进行透视变换,变形

📌参数:

  • src – 输入图像
  • dst – 大小为dsize且类型与src相同的输出图像
  • M – 3×3变换矩阵
  • dsize – 输出图像的大小
  • 下图中红点即为固定顶点,在变换先后固定顶点的像素值不变
  • 图像整体则根据变换规则进行变换同理,透视变换是图像基于4个固定顶点的变换

3.getPerspectiveTransform函数

  • C++方式1:Mat getPerspectiveTransform(InputArray src, InputArray dst)  
  • C++方式2:Mat getPerspectiveTransform(const Point2f src[], const Point2f dst[]) 

📌函数功能:根据源图像和目标图像上的四对点坐标来计算从原图像透视变换到目标头像的透视变换矩阵

📌参数:

  • src – 源图像中四边形顶点的坐标    
  • dst – 目标图像中相应四边形顶点的坐标
  • 该函数计算透视变换的3乘3矩阵,以便实现以下计算:

四、透视变换案例

  • 首先,先试想一下,我们要将下图单独取一张平面出来(例如:A)该如何实现呢? 

案例1:文稿矫正

  • 这是本案例所用到的图片素材

  • 实现效果

  • 其他奇奇怪怪的东西的效果

案例1 完整代码

#include <iostream>
#include <opencv2/opencv.hpp>
using namespace cv;
using namespace std;struct imagedata
{Mat img;//目标图像vector <Point2f> points;//3D点
};//鼠标操作函数:用于选择四个角的点(使用方法有顺序的,从左上角顺时针选择,选完之后回车)
void mouseHundle(int event,int x,int y,int flag,void *per)
{struct imagedata * d=(struct imagedata*)per;if(event==EVENT_LBUTTONDOWN){//确定按下的是鼠标左键//用圆形标记一下鼠标按下左键标记的位置circle(d->img,Point(x,y),3,Scalar(0,255,0),3,CV_AA);//在图上标记,圆心为点击的位置imshow("image",d->img);//原窗口上显示if(d->points.size()<4){d->points.push_back(Point2f(x,y));//把点击的点存起来}}
}void example_1()
{Mat image=imread("C:/Users/86177/Desktop/image/777.jpg");Mat result=Mat::zeros(400,500,CV_8UC1);//400*500的大小,但是里面没有东西//存放四个转换以后的坐标vector <Point2f>obj;obj.push_back(Point2f(0,0));obj.push_back(Point2f(500,0));obj.push_back(Point2f(500,400));obj.push_back(Point2f(0,400));//转换后的坐标imshow("image",image);struct imagedata data;data.img=image;setMouseCallback("image",mouseHundle,&data);//鼠标处理的回调函数waitKey(0);//按任意键关闭当前显示的窗口,显示下一个窗口Mat res=findHomography(data.points,obj,CV_RANSAC);//利用RANSAC算法计算出来一个小矩阵warpPerspective(image,result,res,result.size());  //结果转换imshow("result",result);waitKey(0);
}int main(int argc, char *argv[])
{example_1();return 0;
}

案例2:广告屏换图

  • 这是本案例所用到的图片素材 

  • 实现效果

  • 变换过程,如下图所示: 

案例2 完整代码

#include <iostream>
#include <opencv2/opencv.hpp>
using namespace cv;
using namespace std;struct imagedata
{Mat img;//目标图像vector <Point2f> points;//3D点
};//鼠标操作函数:用于选择四个角的点(使用方法有顺序的,从左上角顺时针选择,选完之后回车)
void mouseHundle(int event,int x,int y,int flag,void *ptr)
{struct imagedata * d=(struct imagedata*)ptr;if(event==EVENT_LBUTTONDOWN){//确定按下的是鼠标左键//用圆形标记一下鼠标按下左键标记的位置circle(d->img,Point(x,y),3,Scalar(0,255,0),3,CV_AA);//在图上标记,圆心为点击的位置//imshow("image",d->img);//原窗口上显示imshow("dst",d->img);//原窗口上显示if(d->points.size()<4)//只存下来,最先点的前四个点{d->points.push_back(Point2f(x,y));//把鼠标操作点击的点存起来}}
}void example_2()
{Mat image1=imread("C:/Users/86177/Desktop/image/2222.jpg");//广告屏替换图片路径Mat image2=imread("C:/Users/86177/Desktop/image/city.jpg");//背景图片路径Mat dst=image2.clone();//克隆vector <Point2f>obj;obj.push_back(Point2f(0,0));obj.push_back(Point2f(image1.cols,0));obj.push_back(Point2f(image1.cols,image1.rows));obj.push_back(Point2f(0,image1.rows));//转换后的坐标imshow("dst",dst);struct imagedata data;data.img =dst;setMouseCallback("dst",mouseHundle,&data);//鼠标处理的回调函数waitKey(0);//按任意键关闭当前显示的窗口,显示下一个窗口Mat res=findHomography(obj,data.points,CV_RANSAC);//利用RANSAC算法,3*3变换矩阵warpPerspective(image1,dst,res,dst.size());//透视转换imshow("warpPerspective",dst);Point pts[4];for(int i=0;i<4;i++){pts[i]=data.points[i];}fillConvexPoly(image2,pts,4,Scalar(0),CV_AA);image2+=dst;imshow("final",image2);waitKey(0);
}int main(int argc, char *argv[])
{example_2();return 0;
}

PS: 案例代码使用说明,需要选择四个点,且必须顺时针选择,因为代码中只设置存储4个点,超出选择无效,选择完毕后按 Enter 即可显示效果

五、总结

  • 本文核心内容包括:OpenCV透视变化概念透视变化工作原理
  • 广告屏换图文稿矫正实现为例子,讲解了OpenCV透视变化的实战案例,案例完成的效果还是挺有意思的!
  • 在我们的日常生活中,已经有很多软件用到了我们的透视变换的知识,例如,我们办公扫描文件时用的扫描软件,很大程度上,节省了我们的时间,提高了工作效率!

以上就是本文的全部内容啦!如果对您有帮助,麻烦点赞啦!收藏啦!欢迎各位评论区留言!! !

这篇关于【OpenCV】广告屏换图 文稿矫正 透视变换 C++ 案例实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/420014

相关文章

使用Python和OpenCV库实现实时颜色识别系统

《使用Python和OpenCV库实现实时颜色识别系统》:本文主要介绍使用Python和OpenCV库实现的实时颜色识别系统,这个系统能够通过摄像头捕捉视频流,并在视频中指定区域内识别主要颜色(红... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间详解

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

PostgreSQL中MVCC 机制的实现

《PostgreSQL中MVCC机制的实现》本文主要介绍了PostgreSQL中MVCC机制的实现,通过多版本数据存储、快照隔离和事务ID管理实现高并发读写,具有一定的参考价值,感兴趣的可以了解一下... 目录一 MVCC 基本原理python1.1 MVCC 核心概念1.2 与传统锁机制对比二 Postg

SpringBoot整合Flowable实现工作流的详细流程

《SpringBoot整合Flowable实现工作流的详细流程》Flowable是一个使用Java编写的轻量级业务流程引擎,Flowable流程引擎可用于部署BPMN2.0流程定义,创建这些流程定义的... 目录1、流程引擎介绍2、创建项目3、画流程图4、开发接口4.1 Java 类梳理4.2 查看流程图4

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

OpenCV实现实时颜色检测的示例

《OpenCV实现实时颜色检测的示例》本文主要介绍了OpenCV实现实时颜色检测的示例,通过HSV色彩空间转换和色调范围判断实现红黄绿蓝颜色检测,包含视频捕捉、区域标记、颜色分析等功能,具有一定的参考... 目录一、引言二、系统概述三、代码解析1. 导入库2. 颜色识别函数3. 主程序循环四、HSV色彩空间

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取

基于Python实现一个Windows Tree命令工具

《基于Python实现一个WindowsTree命令工具》今天想要在Windows平台的CMD命令终端窗口中使用像Linux下的tree命令,打印一下目录结构层级树,然而还真有tree命令,但是发现... 目录引言实现代码使用说明可用选项示例用法功能特点添加到环境变量方法一:创建批处理文件并添加到PATH1