Opencv_10 图像的透视变换

article/2025/10/3 19:47:43

文章目录

      • 一. 透视变换的原理
      • 二. 透视变换实现
        • ① 函数原型
        • ② 透视变换实现
      • 三. 透视变换的案例

一. 透视变换的原理

透视变换(Perspective Transformation)是将图片投影到一个新的视平面(Viewing Plance),也称为投影映射(Projective Mapping).通用的变换公式为:


透视变换矩阵

(x,y,1)是原点, [X,Y,Z]是变换后的点位

这是一个二维空间变换到三维空间的转换,因为图像在二维平面,故除以Z

二. 透视变换实现

① 函数原型

Mat getPerspectiveTransform(const Point2f src[], const Point2f dst[], int solveMethod = DECOMP_LU);

参数说明:

  • src[]: 原图像中的四个像素坐标
  • dst[]: 目标图像中的四个像素点坐标
  • solveMethod: 选择计算透视变换矩阵方法的标志,可以选择参数以及含义在下表中


该函数两个输入量都是存放浮点坐标的数组,在生成数组的时候像素点的输入顺序没有要求,但是需要注意像素点的对应关系是一一对应的,函数的返回值是一个3*3的矩阵.最后一个参数是对应点坐标计算透视变换矩阵方法的选择标志,默认使用的是最佳主轴元素的高斯消元法DECOMP_LU

获取到透视变换矩阵以后,就可以进行透视变换了

void warpPerspective( InputArray src, OutputArray dst,InputArray M, Size dsize,int flags = INTER_LINEAR,int borderMode = BORDER_CONSTANT,const Scalar& borderValue = Scalar());

参数说明:

  • src: 输入图像
  • dst: 输出图像
  • M: 3*3的变换矩阵
  • dSize: 输出图像的尺寸
  • flags: 插值方法标志
  • boardMode: 像素边界填充方式的标志
  • boardValue: 填充边界使用的数值,默认情况下是0

② 透视变换实现

#include"MyOpencv.h"
#include <vector>int main(void)
{Mat imageSrc = Mat::zeros(Size(800, 800), CV_8UC3);// 画一个多边形vector<Point> pts;pts.push_back(Point(50, 40));pts.push_back(Point(600, 40));pts.push_back(Point(600, 600));pts.push_back(Point(50, 600));polylines(imageSrc, pts, true, Scalar(0, 255, 0), 3, LINE_8);pts.clear();pts.push_back(Point(150, 80));pts.push_back(Point(350, 80));pts.push_back(Point(350, 200));pts.push_back(Point(150, 200));polylines(imageSrc, pts, true, Scalar(0, 0, 255), 3, LINE_8);Point2f srcPoint[4];Point2f dstPoint[4];srcPoint[0] = Point2f(150, 80);srcPoint[1] = Point2f(350, 80);srcPoint[2] = Point2f(350, 200);srcPoint[3] = Point2f(150, 200);dstPoint[0] = Point2f(150, 80);dstPoint[1] = Point2f(350, 80);dstPoint[2] = Point2f(400, 200);dstPoint[3] = Point2f(200, 200);Mat M = getPerspectiveTransform(srcPoint, dstPoint);Mat dst;warpPerspective(imageSrc, dst, M, Size(1000, 1000));imshow("Original", imageSrc);imshow("Dst", dst);waitKey(0);return 0;
} 

结果:

三. 透视变换的案例


将这张图像摆正.用到的技术.

  1. 灰度处理,二值化,形态学操作行程连通域
  2. 轮廓发现,将目标的轮廓绘制出来
  3. 在绘制的轮廓中进行直线检测
  4. 找出4个边,求出四个交点
  5. 使用透视变换函数,得到结果
#include "MyOpencv.h"
#include <vector>int main(void)
{Mat imageSrc = imread("./test_01.png", IMREAD_GRAYSCALE);imshow("Original", imageSrc);// 二值化Mat binary;threshold(imageSrc, binary, 0, 255, THRESH_BINARY | THRESH_OTSU);imshow("Binary", binary);// 形态学 开操作,可以填充白色小区域Mat morphImage;Mat kernel = getStructuringElement(MORPH_RECT, Size(5, 5), Point(-1, -1));morphologyEx(binary, morphImage, MORPH_OPEN, kernel, Point(-1, -1), 3);imshow("Opened", morphImage);// 查找轮廓vector<vector<Point>> contours;vector<Vec4i> hireachy;findContours(morphImage, contours, hireachy, RETR_LIST, CHAIN_APPROX_SIMPLE);cout << "Contours.size: " << contours.size() << endl;// 轮廓绘制int width = imageSrc.cols;int height = imageSrc.rows;Mat drawImage = Mat::zeros(imageSrc.size(), CV_8UC3);for (int index = 0; index < contours.size(); index++){Rect rect = boundingRect(contours[index]);if (rect.width > width / 2 && rect.height > height / 2 && rect.width < width - 5 && rect.height < height - 5){drawContours(drawImage, contours, index, Scalar(0, 0, 255), 2, 8,hireachy, 0, Point(0, 0));}}imshow("Contours", drawImage);// 直线检测vector<Vec4i> lines;Mat contoursImage;int accu = static_cast<int>(min(width * 0.3, height * 0.3));cvtColor(drawImage, contoursImage, COLOR_BGR2GRAY);imshow("Contours", contoursImage);Mat linesImage = Mat::zeros(imageSrc.size(), CV_8UC3);HoughLinesP(contoursImage, lines, 1, CV_PI / 180.0,150,accu,accu);for (int index = 0; index < lines.size(); index++){Vec4i ln = lines[index];line(linesImage, Point(ln[0], ln[1]), Point(ln[2], ln[3]),Scalar(0, 0, 255), 2, 8, 0);}cout << "Number of lines: " << lines.size() << endl;imshow("ImageLines",linesImage);// 先确定上下左右四条直线,然后求交点Vec4i topLine, bottomLine, leftLine, rightLine;int yOffset;int xOffset;for (int i = 0; i < lines.size(); i++){Vec4i line = lines[i];yOffset = abs(line[3] - line[1]);xOffset = abs(line[0] - line[2]);// topLine and bottomLineif (yOffset < height * 0.2 ){if (line[1] < height / 2 && line[3] < height / 2){topLine = lines[i];}else{bottomLine = lines[i];}}// leftLine and rightLineif (xOffset < width * 0.2){if (line[0] < width / 2 && line[1] < width / 2){leftLine = lines[i];}else{rightLine = lines[i];}}}cout << "topLine : p1(x,y)= " << topLine[0] << "," << topLine[1] << "; p2(x,y)= " << topLine[2] << "," << topLine[3] << endl;cout << "bottomLine : p1(x,y)= " << bottomLine[0] << "," << bottomLine[1] << "; p2(x,y)= " << bottomLine[2] << "," << bottomLine[3] << endl;cout << "leftLine : p1(x,y)= " << leftLine[0] << "," << leftLine[1] << "; p2(x,y)= " << leftLine[2] << "," << leftLine[3] << endl;cout << "rightLine : p1(x,y)= " << rightLine[0] << "," << rightLine[1] << "; p2(x,y)= " << rightLine[2] << "," << rightLine[3] << endl;// 获取两条直线的交点Point leftTop, rightTop, rightBottom, leftBottom;float leftLineK, leftLineB, topLineK, topLineB, rightLineK, rightLineB, bottomLineK, bottomLineB;leftLineK = float(leftLine[3] - leftLine[1]) / float(leftLine[2] - leftLine[0]);leftLineB = leftLine[1] - leftLineK * leftLine[0];rightLineK = float(rightLine[3] - rightLine[1]) / float(rightLine[2] - rightLine[0]);rightLineB = rightLine[1] - rightLineK * rightLine[0];topLineK = float(topLine[3] - topLine[1]) / float(topLine[2] - topLine[0]);topLineB = topLine[1] - topLineK * topLine[0];bottomLineK = float(bottomLine[3] - bottomLine[1]) /float(bottomLine[2] - bottomLine[0]);bottomLineB = bottomLine[1] - bottomLineK * bottomLine[0];cout << "leftLineK = " << leftLineK << ", leftLineB = " << leftLineB << endl;cout << "rightLineK = " << rightLineK << ", rightLineB = " << rightLineB << endl;cout << "topLineK = " << topLineK << ", topLineB = " << topLineB << endl;cout << "bottomLineK = " << bottomLineK << ", bottomLineB = " << bottomLineB << endl;// 获取交点Point p1;p1.x = -static_cast<int>((leftLineB - topLineB) / (leftLineK - topLineK));p1.y = static_cast<int>(leftLineK * p1.x + leftLineB);Point p2;p2.x = -static_cast<int>((topLineB - rightLineB) / (topLineK - rightLineK));p2.y = static_cast<int>(topLineK * p2.x + topLineB);Point p3;p3.x = -static_cast<int>((rightLineB - bottomLineB) / (rightLineK - bottomLineK));p3.y = static_cast<int>(rightLineK * p3.x + rightLineB);Point p4;p4.x = -static_cast<int>((bottomLineB - leftLineB) / (bottomLineK - leftLineK));p4.y = static_cast<int>(bottomLineK * p4.x + bottomLineB);cout << "Point p1: (" << p1.x << "," << p1.y << ")" << endl;cout << "Point p2: (" << p2.x << "," << p2.y << ")" << endl;cout << "Point p3: (" << p3.x << "," << p3.y << ")" << endl;cout << "Point p4: (" << p4.x << "," << p4.y << ")" << endl;// 获取两点之间的距离'int newWidth, newHeight;newWidth = sqrt((p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y));newHeight = sqrt((p1.x - p4.x) * (p1.x - p4.x) + (p1.y - p4.y) * (p1.y - p4.y));cout << "newWidth = " << newWidth << ", newHeight = " << newHeight << endl;// 透视变换Point2f srcPts[4];Point2f dstPts[4];srcPts[0] = p1;srcPts[1] = p2;srcPts[2] = p3;srcPts[3] = p4;dstPts[0] = Point(0, 0);dstPts[1] = Point(newWidth, 0);dstPts[2] = Point(newWidth, newHeight);dstPts[3] = Point(0, newHeight);Mat M = getPerspectiveTransform(srcPts, dstPts, 0);Mat dst;warpPerspective(imageSrc, dst, M, Size(newWidth+5, newHeight+5), INTER_LINEAR);imshow("Dst", dst);waitKey(0);return 0;
} 

结果:


http://chatgpt.dhexx.cn/article/svf2Bn9q.shtml

相关文章

【OpenCV】透视变换 仿射变换

目录 一&#xff1a;透视变换基本概念 二&#xff1a;透视变换工作原理 三&#xff1a;findHomography函数 四&#xff1a;warpPerspective函数 五&#xff1a;getPerspectiveTransform函数 一&#xff1a;透视变换基本概念 仿射变换(affine transform)与透视变换(perspec…

逆透视变换详解 及 代码实现(二)

根据 逆透视变换详解 及 代码实现(一)的原理 下面我用车上拍摄的车道图像&#xff0c;采用逆透视变换得到的图像&#xff0c;给出代码前我们先看下处理结果。 首先是原始图像&#xff1a; 下图为逆透视变换图像&#xff1a; 下面说具体的实现吧&#xff01;&#xff01; 一、…

透视变换的实现以及透视变换矩阵的构造

透视变换(Perspective Transformation)是什么,无需多说,但是一个非常惨痛的现实是如果你想知道怎么样才能在编程做,你只能得到一些调用opencv函数的文章,简直有病,谁不会调用函数,我搜索实现肯定是要找代码参考看下是怎么实现算法的,你调用函数还敢叫自己“实现”?那我…

图像处理之_仿射变换与透视变换

1. 仿射变换 1) 用途 旋转 (线性变换)&#xff0c;平移 (向量加)&#xff0e;缩放(线性变换)&#xff0c;错切&#xff0c;反转 2) 方法 仿射变换是一种二维坐标到二维坐标之间的线性变换&#xff0c;它保持了二维图形的“平直性”&#xff08;直线经过变换之后依然是直…

Matlab 透视变换 原理及其代码实现

一、透视变换内涵 透视变换本质&#xff1a;将一个图像投影到新的视平面 透视变换思路&#xff1a; 将二维坐标系转换为三维坐标系。 将三维坐标系投影到新的二维坐标系。 该过程属于非线性变换过程&#xff0c;一个菱形在经过非线性变换后得到一个四边形&#xff0c;但是…

art上的透视变换

透视变换 透视变换在比赛中非常重要&#xff0c;我们在地图识别与微调时用到了透视变换。 1. 微调上的使用 在车抵达目标板附件时&#xff0c;摄像头检测到目标板后&#xff0c;可以通过透视变换确定图片距车的位置。这里需要强调下&#xff0c;我们使用的透视变换与四轮使用…

透视变换(Perspective Transform)

**透视变换(Perspective Transform)**是将图片投影到一个新的视角或平面&#xff0c;变换公式如下。 在之前的章节中我们学习过仿射变换(affine transform) 可以看到&#xff0c;而射变换的变换矩阵是3x2的变换矩阵。和仿射变换不同的是&#xff0c;透视变换的变换矩阵是一个3…

仿射变换与透视变换区别

仿射变换 1) 用途 旋转 (线性变换)&#xff0c;平移 (向量加)&#xff0e;缩放(线性变换)&#xff0c;错切&#xff0c;反转 2) 方法 仿射变换是一种二维坐标到二维坐标之间的线性变换&#xff0c;它保持了二维图形的“平直性”&#xff08;直线经过变换之后依然是直线&…

用OpenCV进行透视变换

1. 引言 欢迎回来&#xff01;今天我们将焦点聚焦在我在图像处理中最喜欢的话题之一——透视变换。使用该技术&#xff0c;可以灵活方便的实现各种各样好玩的特效。 闲话少说&#xff0c;我们直接开始吧&#xff01; 2. 单应矩阵 我们首先展开对单应矩阵的深入研究。作为图…

透视变换矩阵详解

原文&#xff1a; OpenGLProjection Matrix (songho.ca) 看这篇文章主要是因为对learnopengl深度测试这一章的些许疑惑&#xff0c; 为什么在片段着色器中&#xff0c;存储的每一个fragment的深度值并不是线性分布&#xff1f;体现在想要显示出场景里面所有物体的深度值&#x…

【图像处理】透视变换 Perspective Transformation

透视变换(Perspective Transformation)是将图片投影到一个新的视平面(Viewing Plane)&#xff0c;也称作投影映射(Projective Mapping)。通用的变换公式为&#xff1a; u,v是原始图片左边&#xff0c;对应得到变换后的图片坐标x,y,其中。变换矩阵可以拆成4部分&#xff0c;表示…

Opencv中的透视变换

唐宇迪视频学习笔记 def four_point_transform(image, pts):# 获取输入坐标点rect order_points(pts)(tl, tr, br, bl) rect# 计算输入的w和h值widthA np.sqrt(((br[0] - bl[0]) ** 2) ((br[1] - bl[1]) ** 2))widthB np.sqrt(((tr[0] - tl[0]) ** 2) ((tr[1] - tl[1])…

OpenCV中的透视变换介绍

点击上方“小白学视觉”&#xff0c;选择加"星标"或“置顶” 重磅干货&#xff0c;第一时间送达本文转自&#xff1a;opencv学堂 透视变换原理 透视变换是将图像从一个视平面投影到另外一个视平面的过程&#xff0c;所以透视变换也被称为投影映射&#xff08;Project…

仿射变换和透视变换

透视变换 1. 欧式变换2. 相似变换3. 仿射变换4. 射影变换/透视变换概念OpenCV 实现 3D 空间存在集中变换方式。 1. 欧式变换 欧式变换保持了向量的长度和夹角&#xff0c;相当于我们把一个刚体原封不动地进行了移动或旋转&#xff0c;不改变它自身的样子&#xff0c;其矩阵表…

基于OpenCV的图像透视变换详解(从理论到实现再到实践)

一、仿射变换与透视变换 一直无法理解两种仿射变换与透视变换的区别&#xff0c;因此详细学习了两种变换的具体细节&#xff0c;重新书写了公式&#xff0c;并给出自己的一些看法。 1.仿射变换 可以认为&#xff0c;仿射变换是透视变换的一种特例。 仿射变换是一种二维坐标到二…

透视变换(perspective transformation)

透视变换(Perspective Transformation)是将图片投影到一个新的视平面(Viewing Plane)&#xff0c;也称作投影映射(Projective Mapping)。如下图所示 透视变换 透视变换的原理和公式推导见如下博客 【图像处理】透视变换 Perspective Transformation 【OpenCV】透视变换 Perspe…

OpenCV 透视变换

OpenCV 透视变换 1. 简介2. 仿射变换2.1. 平移2.2. 旋转2.3. 放缩2.4. 错切2.5. 仿射变换 3. 透视变换 1. 简介 汽车的360度全景影像&#xff0c;从拍照视角变成鸟瞰图 这种变换常常用到透视变换 在了解透视变换前&#xff0c;需要了解一下其他的变换&#xff0c;包括 平移&am…

深入探索透视投影变换

深入探索透视投影变换 最近更新&#xff1a;2013年11月22日 -Twinsen编写 -本人水平有限&#xff0c;疏忽错误在所难免&#xff0c;还请各位数学高手、编程高手不吝赐教 -email: popyynetease.com -B站专栏&#xff1a; https://b23.tv/oWsl6PD 透视投影是3D固定流水线的重要组…

图像透视变换原理及实现

上篇博客讲解了图像的仿射变换原理及实现&#xff0c;这篇博客讲讲透视变换的原理和实现&#xff0c;透视变换也叫投影变换&#xff0c;仿射变换是透视变换的特例。主要是透视变换能保持“直线性”&#xff0c;即原图像里面的直线&#xff0c;经透视变换后仍为直线。下面给出数…

(十四)透视变换

透视变换(Perspective Transformation) 一 图像变换与平面坐标系的关系 1、旋转&#xff1a; 2、平移&#xff1a; 3、刚体变换 4、仿射变换 5、投影变换&#xff08;单应性变换&#xff09; 6 总结一下&#xff1a; 1、刚体变换&#xff1a;平移旋转&#xff0c;只改变物体…