OpenCV-19图像的仿射变换

news2024/11/17 22:44:50

放射变换是图像旋转,缩放,平移的总称,具体的做法是通过一个矩阵和原图片坐标进行计算,得到新的坐标,完成变换,所以关键就是这个矩阵。

一、仿射变换之图像平移

使用API------warpAffine(src ,M, dsize, flags, mode, value)

warp:弯曲             affine:仿射

其中src为图片

M:变换矩阵

dsize:输出图片大小

flag: 与resize中的插值算法一直

mode:边界外推法标志

value:填充边界值

平移矩阵:矩阵中的每个像素由(x,y)组成(x,y)表示这个像素的坐标,假设沿x轴平移tx,沿y轴平移ty,那么最后得到的坐标为(x,y) =  (x + tx, y + ty),用矩阵表示就是:

第一个坐标代表列,第二个坐标代表行。

示例代码如下:

import cv2
import numpy as np


dog = cv2.imread("dog.png")
h, w, ch = dog.shape
print(dog.shape)   # 先行后列

# 写下变换矩阵,最少是float32位
M = np.float32([[1, 0, 200], [0, 1, 0]])  # 第一个对应水平平移,第二个对应上下平移
new_dog = cv2.warpAffine(dog, M,  dsize=(w, h))   # 先列后行

cv2.imshow("dpg", dog)
cv2.imshow("new_dpg", new_dog)
cv2.waitKey(0)
cv2.destroyAllWindows()

输出结果如下:

二、仿射变换之获取变换矩阵

第一中获取变换方程的方法

OpenCV提供了计算变换矩阵的API---getRotationMatrix2D(center, angle, scale)

center: 中心点,以图片的哪个点作为旋转时的中心点

angle:旋转的角度,按照逆时针旋转。

scale:缩放比例,即想图片进行什么样的缩放

示例代码如下:

# M = cv2.getRotationMatrix2D((100, 100), 15, 1)   # 与原图无关 设定中心点进行旋转
M = cv2.getRotationMatrix2D((w/2, h/2), 15, 1)   # 按照图片的中心点进行旋转
new_dog = cv2.warpAffine(dog, M,  dsize=(w, h))   # 先列后行

输出结果如下:

第二种获取变换方程的方法

getAffineTransform(src[], dst[])通过三点可以确定变换后的位置,相当于解方程,3个点对应三个方程,能解出便宜的参数和旋转的角度。

相当于原图的三个点坐标变为另外三个点的坐标,图像按照坐标的格式旋转。

示例代码如下:

sre = np.float32([[200, 100], [300, 100], [200, 300]])   # 初始三个点的坐标
dst = np.float32([[100, 50], [150, 100], [100, 300]])     # 变化后三个点的坐标
M = cv2.getAffineTransform(sre, dst)
new_dog = cv2.warpAffine(dog, M, dsize=(w, h))  # 先列后行

输出结果如下:

三、仿射变换之透视变化

透视变化就是将一种坐标系变为另一种坐标系,简单来说可以把一张“斜”的图变“正”。

使用API---warpPerspective(img,M, dsize)

对于透视变换来说,M是一个3*3的矩阵。

同时使用API---getPerspectiveTransform(src, dst)获取透视变换的变换矩阵,需要4个点,即图片的四个角。

最后在通过cv.namedWindow对窗口进行缩放

示例代码如下:

src = np.float32([[100, 200], [500, 200], [100, 600], [500, 600]])    # 原图的四个坐标
dst = np.float32([[0, 0], [500, 0], [0, 300], [500, 300]])
M = cv2.getPerspectiveTransform(src, dst)
new_dog = cv2.warpPerspective(dog, M, (500, 300))
# 创建窗口对图片进行缩放
cv2.namedWindow("dog", cv2.WINDOW_NORMAL)
cv2.resizeWindow("dog", 640, 480)
cv2.imshow("dog", dog)

cv2.namedWindow("new_dog", cv2.WINDOW_NORMAL)
cv2.resizeWindow("new_dog", 640, 480)
cv2.imshow("new_dog", new_dog)

输出结果如下:

综合演示代码如下所示:

import cv2
import numpy as np

dog = cv2.imread("dog.png")
h, w, ch = dog.shape
print(dog.shape)  # 先行后列

# 写下变换矩阵,最少是float32位
# M = np.float32([[1, 0, 200], [0, 1, 0]])  # 第一个对应水平平移,第二个对应上下平移
# new_dog = cv2.warpAffine(dog, M,  dsize=(w, h))   # 先列后行

# 获取变换矩阵
# M = cv2.getRotationMatrix2D((100, 100), 15, 1)   # 与原图无关 设定中心点进行旋转
# M = cv2.getRotationMatrix2D((w / 2, h / 2), 15, 1)  # 按照图片的中心点进行旋转
# new_dog = cv2.warpAffine(dog, M, dsize=(w, h))  # 先列后行

# 通过三个点的坐标获取变换矩阵
# sre = np.float32([[200, 100], [300, 100], [200, 300]])   # 初始三个点的坐标
# dst = np.float32([[100, 50], [150, 100], [100, 300]])     # 变化后三个点的坐标
# M = cv2.getAffineTransform(sre, dst)
# new_dog = cv2.warpAffine(dog, M, dsize=(w, h))  # 先列后行

# 透视变换
src = np.float32([[100, 200], [500, 200], [100, 600], [500, 600]])    # 原图的四个坐标
dst = np.float32([[0, 0], [500, 0], [0, 300], [500, 300]])
M = cv2.getPerspectiveTransform(src, dst)
new_dog = cv2.warpPerspective(dog, M, (500, 300))

# 创建窗口对图片进行缩放
cv2.namedWindow("dog", cv2.WINDOW_NORMAL)
cv2.resizeWindow("dog", 640, 480)
cv2.imshow("dog", dog)

cv2.namedWindow("new_dog", cv2.WINDOW_NORMAL)
cv2.resizeWindow("new_dog", 640, 480)
cv2.imshow("new_dog", new_dog)
cv2.waitKey(0)
cv2.destroyAllWindows()

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

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

相关文章

Nightingale 夜莺监控系统 - 监控篇(2)

Author:rab 官方文档:https://flashcat.cloud/docs/content/flashcat-monitor/categraf/3-configuration/ 目录 前言一、Categraf 配置文件二、Input 插件配置文件2.1 插件说明2.2 通用配置2.2.1 配置采集频率 interval2.2.2 配置采集实例 instances2.2…

C#编程-在线程中使用同步

在线程中使用同步 在线程应用程序中,线程需要相互共享数据。但是,应用程序应该确保一个线程不更改另一个线程使用的数据。考虑有两个线程的场景。一个线程从文件读取工资,另一个线程尝试更新工资。当两个线程同时工作时,数据就会受损。下图显示了两个线程同时访问一个文件…

【JAVA】concurrentHashMap和HashTable有什么区别

🍎个人博客:个人主页 🏆个人专栏:JAVA ⛳️ 功不唐捐,玉汝于成 目录 前言 正文 同步性质: 性能: 允许空键值(Allow Nulls): 迭代器(Iter…

Flask+ Dependency-injecter+pytest 写测试类

最近在使用这几个在做项目,因为第一次用这个,所以不免有些问题。总结下踩的坑 1.测试类位置 首先测试类约定会放在tests里面,不然有可能发生引入包的问题,会报错某些包找不到。 2. 测试类依赖注入 这里我就用的真实的数据库操作…

[AutoSar]BSW_OS 01 Autosar OS入门(一)

目录 关键词平台说明一、Autosar OS 的位置二、Autosar OS 与OSEK三、TASK 关键词 嵌入式、C语言、autosar、OS、BSW 平台说明 项目ValueOSautosar OSautosar厂商vector芯片厂商TI编程语言C,C编译器HighTec (GCC) 一、Autosar OS 的位置 如在[AutoSar]基础部分 a…

如何使用统计鸟网站统计分析网站流量来源?

统计鸟官网地址:https://www.tongjiniao.com/ 站长必备!网站数据统计,流量监测平台 提供网站数据统计分析、搜索关键词、流量访问来源等服务 深入分析用户点击习惯,为智能化运营网站提供更好的用户体验 目录 一、注册账号信息 二…

电位器的基本知识

一、电位器简介 电位器是一种可调的电子元件。它是由一个电阻体和一个转动或滑动系统组成。当电阻体的两个固定触电之间外加一个电压时,通过转动或滑动系统改变触点在电阻体上的位置,在动触点与固定触点之间便可得到一个与动触点位置成一定关系的电压。…

DFT中的SCAN、BIST、ATPG基本概念

DFT中的SCAN、BIST、ATPG基本概念 SCAN 定义 扫描路径法是一种针对时序电路芯片的DFT方案,目标是在不影响正常功能的情况下来能够提高可控性和可观测性。 原理 原理是将时序电路可以模型化为一个组合电路网络和带触发器(Flip-Flop,简称FF)的时序电路…

【开源】基于JAVA的数据可视化的智慧河南大屏

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块三、系统展示四、核心代码4.1 数据模块 A4.2 数据模块 B4.3 数据模块 C4.4 数据模块 D4.5 数据模块 E 五、免责说明 一、摘要 1.1 项目介绍 基于JAVAVueSpringBootMySQL的数据可视化的智慧河南大屏,包含了GDP、…

蚁群算法(ACO)解决旅行商(TSP)问题的python实现

TSP问题 旅行商问题(Travelling Salesman Problem, 简记TSP,亦称货郎担问题):设有n个城市和距离矩阵D [dij],其中dij表示城市i到城市j的距离,i, j 1, 2 … n,则问题是要找出遍访每个城市恰好一次的一条回…

c#多线程中使用SemaphoreSlim

SemaphoreSlim是一个用于同步和限制并发访问的类,和它类似的还有Semaphore,只是SemaphoreSlim更加的轻量、高效、好用。今天说说它,以及如何使用,在什么时候去使用,使用它将会带来什么优势。 代码的业务是&#xff1a…

InseRF: 文字驱动的神经3D场景中的生成对象插入

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

UE5 简易MC教程学习心得

https://www.bilibili.com/video/BV12G411J7hV?p13&spm_id_frompageDriver&vd_sourceab35b4ab4f3968642ce6c3f773f85138 ———— 目录 0.摧毁逻辑学习 1.发光材质灯方块 2.封装。想让子类 不更改父类的变量。 3.材质命名习惯。 0.摧毁逻辑学习 达到摧毁的条件…

多国管理中心多语言区块链源码一元夺宝程序仿趣步奕跑/原生计步器/原生人脸识别

前后台分开的,后台是TP3.2的框架了。 目前把整体UI 改版黄色系风格,集成了一元夺宝程序,用户数据同步趣步,效果看起来很棒,另外加入股票走势图(K线图),目前已经继承人脸识别&#xf…

数据结构——二叉树(先序、中序、后序及层次四种遍历(C语言版))超详细~ (✧∇✧) Q_Q

目录 ​​​​​​​ 二叉树的定义: *特殊的二叉树: 二叉树的性质: 二叉树的声明: 二叉树的先序遍历: 二叉树的中序遍历: 二叉树的后序遍历: 二叉树的层序遍历: 二叉树的节…

【工具使用-A2B】32通道24bit传输的配置方法

一,简介 本文记录A2B总线上压缩为24bit的方法和如何计算是否超带宽的方法。 二,具体操作 2.1 计算是否超带宽的方法 确认传输带宽占比 2.2 32通道24bit配置 Downstream Compression选择Disable。 三,总结 本文主要介绍,如…

如何在Spring Boot中使用EhCache缓存

1、EhCache介绍 在查询数据的时候,数据大多来自于数据库,我们会基于SQL语句与数据库交互,数据库一般会基于本地磁盘IO将数据读取到内存,返回给Java服务端,我们再将数据响应给前端,做数据展示。 但是MySQL…

红队专题-Golang工具ChYing

Golang工具ChYing 招募六边形战士队员原chying工具代码分析并发访问控制并发 原子 写入读取 通道嵌套映射结构初始化启动代理服务器重启代理服务器 招募六边形战士队员 一起学习 代码审计、安全开发、web攻防、逆向等。。。 私信联系 原chying工具代码分析 前有 Chying 后有…

什么是云服务器,阿里云优势如何?

阿里云服务器ECS英文全程Elastic Compute Service,云服务器ECS是一种安全可靠、弹性可伸缩的云计算服务,阿里云提供多种云服务器ECS实例规格,如经济型e实例、通用算力型u1、ECS计算型c7、通用型g7、GPU实例等,阿里云百科aliyunbai…

C++|44.智能指针

文章目录 智能指针unique_ptr特点一——无法进行复制 shared_ptr特点一——可复制特点二——计数器(用于确定删除的时机) 其他 智能指针 通常的指针是需要特殊地去申请对应的空间,并在不使用的时候还需要人工去销毁。 而智能指针相对普通的指…