DataWhale 机器视觉组队学习task2

2.1 简介

  该部分将对基本的几何变换进行学习,几何变换的原理大多都是相似,只是变换矩阵不同,因此,我们以最常用的平移和旋转为例进行学习。在深度学习领域,我们常用平移、旋转、镜像等操作进行数据增广;在传统CV领域,由于某些拍摄角度的问题,我们需要对图像进行矫正处理,而几何变换正是这个处理过程的基础,因此了解和学习几何变换也是有必要的。

这次我们带着几个问题进行,以旋转为例:

  • 1:变换的形式(公式)是什么?

  • 2:旋转中心是什么?毕竟以不同位置为旋转中心得到的结果是不一样的。

  • 3:采用前向映射还是反向映射?(反向映射更为有效)

  • 4:采用反向映射后,采用何种插值算法?最常用的的是双线性插值,OpenCV也是默认如此。

2.2 学习目标

  • 了解几何变换的概念与应用

  • 理解平移、旋转的原理

  • 掌握在OpenCV框架下实现平移、旋转操作

2.3 内容介绍

1、平移、旋转的原理

2、OpenCV代码实践

3、动手实践并打卡(读者完成)

2.4 算法理论介绍

2.4.1 向前映射与向后映射

前向映射:

  图像的几何变换就是建立一种源图像像素与变换后的图像像素之间的映射关系。也正是通过这种映射关系可以知道原图像任意像素点变换后的坐标,或者是变换后的图像在原图像的坐标位置等。

  用简单的数学公式可以表示为:

(xy)=(U(u,v)V(u,v))f(u,v)=(x,y)\begin{pmatrix}x\\y\end{pmatrix}=\begin{pmatrix}U(u,v)\\V(u,v)\end{pmatrix}\\ f(u,v)=(x,y)(xy​)=(U(u,v)V(u,v)​)f(u,v)=(x,y)

  其中,xxx,yyy代表输出图像像素的坐标,uuu,vvv表示输入图像的像素坐标,而UUU,VVV表示的是两种映射关系,fff是将点(u,v)(u,v)(u,v)映射到(x,y)(x,y)(x,y)的映射关系,需要说明的是,映射关系可以是线性关系,也可以是多项式关系
  从上面的映射关系可以看到,只要给出了图像上任意的像素坐标,都能够通过对应的映射关系获得几何变换后的像素坐标。这种将输入映射到输出的过程我们称之为 “向前映射”。但是在实际应用中,向前映射会出现如下几个问题:

  1. 浮点数坐标,如(1,1)映射为(0.5,0.5),显然这是一个无效的坐标,这时我们需要使用插值算法进行进一步处理。

  2. 可能会有多个像素坐标映射到输出图像的同一位置,也可能输出图像的某些位置完全没有相应的输入图像像素与它匹配,也就是没有被映射到,造成有规律的空洞(黑色的蜂窝状)。

什么是有规律的空洞呢?下面举个例子大家就明白了


  可以从上图知道:原图经过前向映射旋转了30度后,输出图像中有规律的空洞(黑色的蜂窝状),那这些空洞是这么来的呢?

  可以看到,旋转三十度后,输出图像两个红色的点被映射到同一个坐标,而没有点被映射到绿色问号处,这就造成了间隙和重叠,导致出现蜂窝状空洞。

向后映射

  为了克服前向映射的这些不足,因此引进了“后向映射”

它的数学表达式为:

(uv)=(U−1(x,y)V−1(x,y))f−1(x,y)=(u,v)\begin{pmatrix}u\\v\end{pmatrix}=\begin{pmatrix}U^{-1}(x,y)\\V^{-1}(x,y)\end{pmatrix} \\ f^{-1}(x,y)=(u,v)(uv​)=(U−1(x,y)V−1(x,y)​)f−1(x,y)=(u,v)

  可以看出,后向映射与前向映射刚好相反,它是由输出图像的像素坐标反过来推算该像素为在源图像中的坐标位置。这样,输出图像的每个像素值都能够通过这个映射关系找到对应的为止。而不会造成上面所提到的映射不完全和映射重叠的现象。在实际处理中基本上都运用向后映射来进行图像的几何变换。但是反向映射也有一个和前向映射一样的问题, 就是映射后会有小数,需通过插值方法决定输出图像该位置的值,OpenCV默认为双线性插值。

  在使用过程中,如果在一些不改变图像大小的几何变换中,向前映射还是十分有效的,向后映射主要运用在图像的旋转的缩放中,因为这些几何变换都会改变图像的大小。

2.4.1 几何变换

  先看第一个问题,变换的形式。在本篇文章里图像的几何变换全部都采用统一的矩阵表示法,形式如下:

[xy1]=[a0a1a2a3a4a5001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} a_0 &a_1 & a_2 \\ a_3 & a_4 & a_5 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​a0​a3​0​a1​a4​0​a2​a5​1​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

  这就是向前映射的矩阵表示法,其中xxx,yyy表示输出图像像素的坐标,x0x_0x0​,y0y_0y0​表示输入图像像素的坐标,同理,向后映射的矩阵表示为:

[x0y01]=[b0b1b2b3b4b5001][xy1]\begin{bmatrix}x_0 \\y_0\\1\end{bmatrix} =\begin{bmatrix} b_0 &b_1 & b_2 \\ b_3 & b_4 & b_5 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x \\ y \\ 1 \end{bmatrix}⎣⎡​x0​y0​1​⎦⎤​=⎣⎡​b0​b3​0​b1​b4​0​b2​b5​1​⎦⎤​⎣⎡​xy1​⎦⎤​

  可以证明,向后映射的矩阵的表示正好是向前映射的逆变换。

变换名称 a0a_0a0​ a1a_1a1​ a2a_2a2​ a3a_3a3​ a4a_4a4​ a5a_5a5​
平移 1 0 △x\triangle x△x 0 1 △y\triangle y△y
均匀缩放 sss 0 0 0 sss 0
不均匀缩放 sxs_xsx​ 0 0 0 sys_ysy​ 0
顺时针旋转角度θ\thetaθ cosθcos\thetacosθ sinθsin\thetasinθ 0 −sinθ-sin\theta−sinθ cosθcos\thetacosθ 0
逆时针旋转角度θ\thetaθ cosθcos\thetacosθ −sinθ-sin\theta−sinθ 0 sinθsin\thetasinθ cosθcos\thetacosθ 0
垂直偏移变换 1 0 0 h 1 0
水平偏移变换 1 h 0 0 1 0

下面举几个例子
原图:

  1. 向上平移一个单位向右平移一个单位

[xy1]=[101011001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} 1 &0 & 1 \\ 0 & 1 & 1 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​100​010​111​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

  1. 放大为原来的两倍

[xy1]=[200020001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} 2 &0 & 0 \\ 0 & 2 & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​200​020​001​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

  1. 顺时针旋转45度

[xy1]=[2/22/20−2/22/20001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} \sqrt2/2 &\sqrt2/2 & 0 \\ -\sqrt2/2 & \sqrt2/2 & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​2​/2−2​/20​2​/22​/20​001​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

  1. 水平偏移2个单位

[xy1]=[120010001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} 1 & 2 & 0 \\ 0 & 1 & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​100​210​001​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

2.4.2坐标系变换

  再看第二个问题,变换中心,对于缩放、平移可以以图像坐标原点(图像左上角为原点)为中心变换,这不用坐标系变换,直接按照一般形式计算即可。而对于旋转和偏移,一般是以图像中心为原点,那么这就涉及坐标系转换了。

  我们都知道,图像坐标的原点在图像左上角,水平向右为 X 轴,垂直向下为 Y 轴。数学课本中常见的坐标系是以图像中心为原点,水平向右为 X 轴,垂直向上为 Y 轴,称为笛卡尔坐标系。看下图:

因此,对于旋转和偏移,就需要3步(3次变换):

  • 将输入原图图像坐标转换为笛卡尔坐标系;
  • 进行旋转计算。旋转矩阵前面已经给出了;
  • 将旋转后的图像的笛卡尔坐标转回图像坐标。

2.4.3图像坐标系与笛卡尔坐标系转换关系:

先看下图:

  在图像中我们的坐标系通常是AB和AC方向的,原点为A,而笛卡尔直角坐标系是DE和DF方向的,原点为D。
  令图像表示为M×N的矩阵,对于点A而言,两坐标系中的坐标分别是(0,0)和(−N/2,M/2)(-N/2,M/2)(−N/2,M/2),则图像某像素点(x′,y′)(x',y')(x′,y′)转换为笛卡尔坐标(x,y)(x,y)(x,y)转换关系为,xxx为列,yyy为行:

x=x′−N2x=x'-\frac{N}{2}x=x′−2N​

y=−y′−M2y=-y'-\frac{M}{2}y=−y′−2M​

逆变换为:

x′=x+N2x'=x+\frac{N}{2}x′=x+2N​

y′=−y+M2y'=-y+\frac{M}{2}y′=−y+2M​

  于是,根据前面说的3个步骤(3次变换),旋转(顺时针旋转)的变换形式就为,3次变换就有3个矩阵:

[xy1]=[10−0.5⋅N0−1−0.5⋅M001][cosθsinθ0−sinθcosθ0001][100.5⋅N0−10.5⋅M001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} 1&0 & -0.5\cdot N \\ 0 & -1 & -0.5 \cdot M \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} cos\theta&sin\theta & 0 \\ -sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} 1&0 & 0.5\cdot N \\ 0 & -1 & 0.5 \cdot M \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​100​0−10​−0.5⋅N−0.5⋅M1​⎦⎤​⎣⎡​cosθ−sinθ0​sinθcosθ0​001​⎦⎤​⎣⎡​100​0−10​0.5⋅N0.5⋅M1​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

即:

[xy1]=[cosθ−sinθ−0.5N(1−cosθ)+0.5Msinθsinθ−cosθ−0.5M(1−sinθ)−0.5Mcosθ001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} = \begin{bmatrix} cos\theta&-sin\theta & -0.5N(1-cos\theta)+0.5Msin\theta \\ sin\theta & -cos\theta & -0.5M(1-sin\theta)-0.5Mcos\theta \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​cosθsinθ0​−sinθ−cosθ0​−0.5N(1−cosθ)+0.5Msinθ−0.5M(1−sinθ)−0.5Mcosθ1​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

2.5 基于OpenCV的实现

  • 工具:OpenCV4.1.0+VS2019
  • 平台:WIN10

函数原型(c++)

OpenCV仿射变换相关的函数一般涉及到warpAffine和getRotationMatrix2D这两个:

  • 使用OpenCV函数warpAffine 来实现一些简单的重映射.
  • OpenCV函数getRotationMatrix2D 来获得旋转矩阵。

1、warpAffined函数详解

void boxFilter( InputArray src, OutputArray dst, int ddepth,Size ksize,  Point anchor = Point(-1,-1),bool normalize = true,int borderType = BORDER_DEFAULT );
  • 第一个参数,InputArray类型的src,输入图像,即源图像,填Mat类的对象即可。
  • 第二个参数,OutputArray类型的dst,函数调用后的运算结果存在这里,需和源图片有一样的尺寸和类型。
  • 第三个参数,InputArray类型的M,2×3的变换矩阵。
  • 第四个参数,Size类型的dsize,表示输出图像的尺寸。
  • 第五个参数,int类型的flags,插值方法的标识符。此参数有默认值INTER_LINEAR(线性插值),可选的插值方式如下:
    INTER_NEAREST - 最近邻插值
    INTER_LINEAR - 线性插值(默认值)
    INTER_AREA - 区域插值
    INTER_CUBIC –三次样条插值
    INTER_LANCZOS4 -Lanczos插值
    CV_WARP_FILL_OUTLIERS - 填充所有输出图像的象素。如果部分象素落在输入图像的边界外,那么它们的值设定为 fillval.
    CV_WARP_INVERSE_MAP –表示M为输出图像到输入图像的反变换,即 。因此可以直接用来做象素插值。否则, warpAffine函数从M矩阵得到反变换。
  • 第六个参数,int类型的borderMode,边界像素模式,默认值为BORDER_CONSTANT。
  • 第七个参数,const Scalar&类型的borderValue,在恒定的边界情况下取的值,默认值为Scalar(),即0。

2、getRotationMatrix2D函数详解

C++: Mat getRotationMatrix2D(Point2f center, double angle, double scale)

参数:

  • 第一个参数,Point2f类型的center,表示源图像的旋转中心。
  • 第二个参数,double类型的angle,旋转角度。角度为正值表示向逆时针旋转(坐标原点是左上角)。
  • 第三个参数,double类型的scale,缩放系数。

实现示例(c++)

1、旋转

 Mat src = imread("../image/source3.jpg");//读取原图像cv::Mat dst;//旋转角度double angle = 45;cv::Size src_sz = src.size();cv::Size dst_sz(src_sz.height, src_sz.width);int len = std::max(src.cols, src.rows);//指定旋转中心(图像中点)cv::Point2f center(len / 2., len / 2.);//获取旋转矩阵(2x3矩阵)cv::Mat rot_mat = cv::getRotationMatrix2D(center, angle, 1.0);//根据旋转矩阵进行仿射变换cv::warpAffine(src, dst, rot_mat, dst_sz);//显示旋转效果cv::imshow("image", src);cv::imshow("result", dst);cv::waitKey(0);return 0;

但是这么写有一个问题,就是旋转后图像会被截断,如下图所示:(左边为原图,右边为顺时针旋转45度后的图)

可以看到,图像的一部分被截断了,其原因是:

  1. 旋转过后的图像大小应该发生变化才能装下旋转后的图片
  2. OpenCv将坐标转成笛卡尔坐标系后没转回图像坐标系

其中比较难理解的是图像大小的变换,下面举一个例子大家就能明白了:

如图:ABCD是变换前矩形,EFGH是变换后的矩形,变换的矩阵表示为:

[xy1]=[cosθsinθ0−sinθcosθ0001][x0y01]\begin{bmatrix}x \\y\\1\end{bmatrix} =\begin{bmatrix} cos\theta &sin\theta & 0 \\ -sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \end{bmatrix} \begin{bmatrix} x_0 \\ y_0 \\ 1 \end{bmatrix}⎣⎡​xy1​⎦⎤​=⎣⎡​cosθ−sinθ0​sinθcosθ0​001​⎦⎤​⎣⎡​x0​y0​1​⎦⎤​

即表达式为:

x=cosθx0+sinθy0x=cos\theta x_0+sin\theta y_0x=cosθx0​+sinθy0​

y=−sinθx0+cosθy0y=-sin\theta x_0 +cos\theta y_0y=−sinθx0​+cosθy0​

  所以,要算旋转后图片的大小,只需计算原图像四个顶点变换后的图像所确定的外接矩形长宽。因为经过坐标变换后的图像是关于原点对称的,所以计算D点变换后的横坐标的绝对值乘2,就是变换后矩形的长,计算A点变换后的纵坐标的绝对值乘2,就是变换后矩形的宽

设原图像长为2a2a2a,宽为2b2b2b,变换后的图像长宽为ccc,ddd,则AAA点的坐标为:(−a,b)(-a, b)(−a,b), DDD点坐标:(a,b)(a, b)(a,b)

c=2∗(a∣cosθ∣+b∣sinθ∣)c = 2*(a| cos\theta | + b|sin\theta |)c=2∗(a∣cosθ∣+b∣sinθ∣)

d=2∗(a∣sinθ∣+b∣cosθ∣)d = 2 * (a|sin\theta | + b|cos\theta |)d=2∗(a∣sinθ∣+b∣cosθ∣)

 Mat src = imread("../image/source3.jpg");//读取原图像Mat dst;// 旋转角度double angle = 45.0;// 计算旋转后输出图形的尺寸int rotated_width = ceil(src.rows * fabs(sin(angle * CV_PI / 180)) + src.cols * fabs(cos(angle * CV_PI / 180)));int rotated_height = ceil(src.cols * fabs(sin(angle * CV_PI / 180)) + src.rows * fabs(cos(angle * CV_PI / 180)));// 计算仿射变换矩阵Point2f center(src.cols / 2, src.rows / 2);Mat rotate_matrix = getRotationMatrix2D(center, angle, 1.0);// 防止切边,对平移矩阵B进行修改rotate_matrix.at<double>(0, 2) += (rotated_width - src.cols) / 2;rotate_matrix.at<double>(1, 2) += (rotated_height - src.rows) / 2;// 应用仿射变换warpAffine(src, dst, rotate_matrix, Size(rotated_width, rotated_height), INTER_LINEAR, 0, Scalar(255, 255, 255));imshow("result", dst);cv::imwrite("right.jpg", dst);waitKey();return 0;

结果:

2、平移

 Mat src = imread("../image/source2.jpg");//读取原图像cv::Mat dst;cv::Size dst_sz = src.size();//定义平移矩阵cv::Mat t_mat =cv::Mat::zeros(2, 3, CV_32FC1);t_mat.at<float>(0, 0) = 1;t_mat.at<float>(0, 2) = 300; //水平平移量t_mat.at<float>(1, 1) = 1;t_mat.at<float>(1, 2) = 300; //竖直平移量//根据平移矩阵进行仿射变换cv::warpAffine(src, dst, t_mat, dst_sz);//显示平移效果cv::imshow("image", src);cv::imshow("result", dst);cv::waitKey(0);return 0;

3.仿射变化

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std;int main(int argc, char** argv)
{Mat src = imread("../image/source2.jpg");//读取原图像//分别在原图像和目标图像上定义三个点Point2f srcTri[3];Point2f dstTri[3];srcTri[0] = Point2f(0, 0);srcTri[1] = Point2f(src.cols - 1, 0);srcTri[2] = Point2f(0, src.rows - 1);dstTri[0] = Point2f(src.cols * 0.0, src.rows * 0.33);dstTri[1] = Point2f(src.cols * 0.85, src.rows * 0.25);dstTri[2] = Point2f(src.cols * 0.15, src.rows * 0.7);Mat dst;//目标图像//设置目标图像的大小和类型与原图像一致,初始像素值都为0dst = Mat::zeros(src.rows, src.cols, src.type());//计算仿射变换矩阵Mat trans_mat = getAffineTransform(srcTri, dstTri);//对原图像应用上面求得的仿射变换warpAffine(src, dst, trans_mat, src.size());//显示结果imshow("origin_image", src);imshow("dst_image", dst);//储存图像imwrite("dst1.jpg", dst);waitKey(0);return 0;
}

进阶实现(根据原理自己实现)

1、旋转

/*图像旋转(以图像中心为旋转中心)*/
void affine_trans_rotate(cv::Mat& src, cv::Mat& dst, double Angle){double angle = Angle*CV_PI / 180.0;//构造输出图像int dst_rows = round(fabs(src.rows * cos(angle)) + fabs(src.cols * sin(angle)));//图像高度int dst_cols = round(fabs(src.cols * cos(angle)) + fabs(src.rows * sin(angle)));//图像宽度if (src.channels() == 1) {dst = cv::Mat::zeros(dst_rows, dst_cols, CV_8UC1); //灰度图初始} else {dst = cv::Mat::zeros(dst_rows, dst_cols, CV_8UC3); //RGB图初始}cv::Mat T1 = (cv::Mat_<double>(3,3) << 1.0,0.0,0.0 , 0.0,-1.0,0.0, -0.5*src.cols , 0.5*src.rows , 1.0); // 将原图像坐标映射到数学笛卡尔坐标cv::Mat T2 = (cv::Mat_<double>(3,3) << cos(angle),-sin(angle),0.0 , sin(angle), cos(angle),0.0, 0.0,0.0,1.0); //数学笛卡尔坐标下顺时针旋转的变换矩阵double t3[3][3] = { { 1.0, 0.0, 0.0 }, { 0.0, -1.0, 0.0 }, { 0.5*dst.cols, 0.5*dst.rows ,1.0} }; // 将数学笛卡尔坐标映射到旋转后的图像坐标cv::Mat T3 = cv::Mat(3.0,3.0,CV_64FC1,t3);cv::Mat T = T1*T2*T3;cv::Mat T_inv = T.inv(); // 求逆矩阵for (double i = 0.0; i < dst.rows; i++){for (double j = 0.0; j < dst.cols; j++){cv::Mat dst_coordinate = (cv::Mat_<double>(1, 3) << j, i, 1.0);cv::Mat src_coordinate = dst_coordinate * T_inv;double v = src_coordinate.at<double>(0, 0); // 原图像的横坐标,列,宽double w = src_coordinate.at<double>(0, 1); // 原图像的纵坐标,行,高//   std::cout << v << std::endl;/*双线性插值*/// 判断是否越界if (int(Angle) % 90 == 0) {if (v < 0) v = 0; if (v > src.cols - 1) v = src.cols - 1;if (w < 0) w = 0; if (w > src.rows - 1) w = src.rows - 1; //必须要加上,否则会出现边界问题}if (v >= 0 && w >= 0 && v <= src.cols - 1 && w <= src.rows - 1){int top = floor(w), bottom = ceil(w), left = floor(v), right = ceil(v); //与映射到原图坐标相邻的四个像素点的坐标double pw = w - top ; //pw为坐标 行 的小数部分(坐标偏差)double pv = v - left; //pv为坐标 列 的小数部分(坐标偏差)if (src.channels() == 1){//灰度图像dst.at<uchar>(i, j) = (1 - pw)*(1 - pv)*src.at<uchar>(top, left) + (1 - pw)*pv*src.at<uchar>(top, right) + pw*(1 - pv)*src.at<uchar>(bottom, left) + pw*pv*src.at<uchar>(bottom, right);}else{//彩色图像dst.at<cv::Vec3b>(i, j)[0] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[0] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[0] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[0] + pw*pv*src.at<cv::Vec3b>(bottom, right)[0];dst.at<cv::Vec3b>(i, j)[1] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[1] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[1] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[1] + pw*pv*src.at<cv::Vec3b>(bottom, right)[1];dst.at<cv::Vec3b>(i, j)[2] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[2] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[2] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[2] + pw*pv*src.at<cv::Vec3b>(bottom, right)[2];}}}}
}

2、平移

/*平移变换*(以图像左顶点为原点)/
/****************************************
tx: 水平平移距离 正数向右移动 负数向左移动
ty: 垂直平移距离 正数向下移动 负数向上移动
*****************************************/
void affine_trans_translation(cv::Mat& src, cv::Mat& dst, double tx, double ty){//构造输出图像int dst_rows = src.rows;//图像高度int dst_cols = src.cols;//图像宽度if (src.channels() == 1) {dst = cv::Mat::zeros(dst_rows, dst_cols, CV_8UC1); //灰度图初始}else {dst = cv::Mat::zeros(dst_rows, dst_cols, CV_8UC3); //RGB图初始}cv::Mat T = (cv::Mat_<double>(3, 3) << 1,0,0 , 0,1,0 , tx,ty,1); //平移变换矩阵cv::Mat T_inv = T.inv(); // 求逆矩阵for (int i = 0; i < dst.rows; i++){for (int j = 0; j < dst.cols; j++){cv::Mat dst_coordinate = (cv::Mat_<double>(1, 3) << j, i, 1);cv::Mat src_coordinate = dst_coordinate * T_inv;double v = src_coordinate.at<double>(0, 0); // 原图像的横坐标,列,宽double w = src_coordinate.at<double>(0, 1); // 原图像的纵坐标,行,高/*双线性插值*/// 判断是否越界if (v >= 0 && w >= 0 && v <= src.cols - 1 && w <= src.rows - 1){int top = floor(w), bottom = ceil(w), left = floor(v), right = ceil(v); //与映射到原图坐标相邻的四个像素点的坐标double pw = w - top; //pw为坐标 行 的小数部分(坐标偏差)double pv = v - left; //pv为坐标 列 的小数部分(坐标偏差)if (src.channels() == 1){//灰度图像dst.at<uchar>(i, j) = (1 - pw)*(1 - pv)*src.at<uchar>(top, left) + (1 - pw)*pv*src.at<uchar>(top, right) + pw*(1 - pv)*src.at<uchar>(bottom, left) + pw*pv*src.at<uchar>(bottom, right);}else{//彩色图像dst.at<cv::Vec3b>(i, j)[0] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[0] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[0] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[0] + pw*pv*src.at<cv::Vec3b>(bottom, right)[0];dst.at<cv::Vec3b>(i, j)[1] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[1] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[1] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[1] + pw*pv*src.at<cv::Vec3b>(bottom, right)[1];dst.at<cv::Vec3b>(i, j)[2] = (1 - pw)*(1 - pv)*src.at<cv::Vec3b>(top, left)[2] + (1 - pw)*pv*src.at<cv::Vec3b>(top, right)[2] + pw*(1 - pv)*src.at<cv::Vec3b>(bottom, left)[2] + pw*pv*src.at<cv::Vec3b>(bottom, right)[2];}}}}
}

相关技术文档、博客、教材、项目推荐

opencv文档: https://docs.opencv.org/3.1.0/da/d54/group__imgproc__transform.html#ga0203d9ee5fcd28d40dbc4a1ea4451983
博客:https://blog.csdn.net/weixin_40647819/article/details/87912122
https://www.jianshu.com/p/18cd12e776e1
https://blog.csdn.net/whuhan2013/article/details/53814026
python版本:https://blog.csdn.net/g11d111/article/details/79978582
https://www.kancloud.cn/aollo/aolloopencv/264331 http://www.woshicver.com/FifthSection/4_2_%E5%9B%BE%E5%83%8F%E5%87%A0%E4%BD%95%E5%8F%98%E6%8D%A2/

2.6 总结

该部分对几何变换的平移和旋转进行了介绍,读者可根据提供的资料对相关原理进行学习,然后参考示例代码自行实现。另外读者可以尝试学习并实现其他几何变换,如偏移。

关于Datawhale

Datawhale是一个专注于数据科学与AI领域的开源组织,汇集了众多领域院校和知名企业的优秀学习者,聚合了一群有开源精神和探索精神的团队成员。Datawhale以“for the learner,和学习者一起成长”为愿景,鼓励真实地展现自我、开放包容、互信互助、敢于试错和勇于担当。同时Datawhale 用开源的理念去探索开源内容、开源学习和开源方案,赋能人才培养,助力人才成长,建立起人与人,人与知识,人与企业和人与未来的联结。

Opencv图像几何变换相关推荐

  1. OpenCV图像几何变换专题(缩放、翻转、仿射变换及透视)【python-Open_CV系列(五)】

    OpenCV图像几何变换专题(缩放.翻转.仿射变换及透视)(python为工具) [Open_CV系列(五)] 文章目录 准备图片 1. 缩放 cv2.resize()方法 2. 翻转 cv2.fli ...

  2. [Python图像处理] 三十六.OpenCV图像几何变换万字详解(平移缩放旋转、镜像仿射透视)

    该系列文章是讲解Python OpenCV图像处理知识,前期主要讲解图像入门.OpenCV基础用法,中期讲解图像处理的各种算法,包括图像锐化算子.图像增强技术.图像分割等,后期结合深度学习研究图像识别 ...

  3. OpenCV图像几何变换——转置,镜像,倒置

    图像几何变换方法之--remap使用. 源图像 一.图像转置 #include <iostream> #include <opencv2/opencv.hpp> using n ...

  4. OpenCV-Python学习(18)—— OpenCV 图像几何变换之图像平移(cv.warpAffine)

    1. 学习目标 学习图像的平移矩阵: 学习 OpenCV 图像平移函数. 2. 图像的平移矩阵 平移是物体位置在水平和垂直方向的移动. 像素点 (x,y) 沿 x 轴平移 dx.沿 y 轴平移 dy, ...

  5. OpenCV-Python学习(19)—— OpenCV 图像几何变换之图像缩放(cv.warpAffine、cv.resize)

    1. 学习目标 学习图像的缩放矩阵: 学习 OpenCV 图像缩放函数 cv.resize 和 cv.warpAffine. 2. 图像的缩放矩阵 缩放是物体在 x 轴和 y 轴的缩放比例. fx 是 ...

  6. opencv android 透视,OpenCV图像几何变换之透视变换

    本文实例为大家分享了Android九宫格图片展示的具体代码,供大家参考,具体内容如下 1. 基本原理 透视变换(Perspective Transformation)的本质是将图像投影到一个新的视平面 ...

  7. opencv 图像几何变换

    变换 大小 import numpy as np import cv2 as cv img = cv.imread('messi5.jpg') res = cv.resize(img,None,fx= ...

  8. Python 计算机视觉(五)—— OpenCV 进行图像几何变换

    几何变换不改变图像的像素值,只是实现图像像素点的重新安排:恰当的进行图像的几何变换,可以减小甚至避免由于角度等一些因素造成的图像失真问题,有利于我们在识别图像时将注意力集中到图像的有效信息中而不至于被 ...

  9. OpenCV中的图像处理 —— 改变颜色空间+图像几何变换

    OpenCV中的图像处理 -- 改变颜色空间+图像几何变换 这一部分主要介绍OpenCV图像处理中的改变颜色空间和图像的几何变换,颜色空间的改变应用非常广泛,在处理图像的实际问题中,经常需要要图像变换 ...

最新文章

  1. 港中文周博磊:十年之间的CVPR与我们(附CVPR2020部分论文链接/开源代码/解读)...
  2. 产品经理日常表情包大全,多说是泪拿走不谢!
  3. c#生成随机位数的汉字字符串
  4. SQLite中的高级SQL
  5. Android CTS 测试总结【转】
  6. oracle 修改nls_characterset,ORACLE NLS_CHARACTERSET字符集的更改
  7. struts2.1.6教程二、struts.xml配置及例程
  8. setTimeOut传参数
  9. Git部署远程仓库至github
  10. 【字符串】面试题之替换子串
  11. 海康摄像机通过Ehome协议接入EasyCVR无法成功上线的原因排查及配置注意事项
  12. wav音频文件转换为sbc音频文件
  13. python 图片合并pdf_利用python将多张图片合并为pdf文档
  14. 台式计算机怎么开声音,台式机如何设置声音
  15. mysql recordcount 1_移植到 MySQL-对 MYSQL 数据库使用 ASP Recordcount 的问题
  16. 微信小程序中的变量和作用域
  17. 广告营销用户点击预测分析
  18. 竹林蹊径:深入浅出Windows驱动开发(china-pub预订中)
  19. 毕业后的项目经历1-项目名称 HIS微服务 类型 医院系统
  20. dell 安装显卡驱动,重启进不了系统解决方法

热门文章

  1. Linux系统安装 | Docker安装最新版本Ubuntu并启动容器
  2. sizeof是c语言的一种运算符,kingsize是什么意思? C语言中sizeof是什么意思
  3. 2021程序员进阶宝典!工信部java中级软件工程师
  4. 初级平面设计师必须熟练掌握的软件
  5. Citrix Netscaler log client ip address
  6. 科学计算机中括号怎么用,科学网—翻译: 科技写作中括号的用法 - 邱敦莲的博文...
  7. iOS 之 Bonjour 协议简单抓包分析
  8. 西交《生物化学》在线作业
  9. Normal map (Bump mapping) 法线贴图(凹凸映射) Standard Shader系列10
  10. MFC OpenGL标签云 (转)