1. 引言

图像旋转是计算机视觉中最基础也是最重要的几何变换之一,在图像处理、计算机视觉、医学影像分析等领域有着广泛应用。OpenCV作为最流行的计算机视觉库,提供了强大的图像旋转功能。本文将深入探讨OpenCV中的两种旋转方式:基于单点的仿射变换旋转和直接图片旋转,并通过代码示例展示如何实现这些功能。

2. 图像旋转的基本概念

图像旋转是指将图像围绕某个点(通常是中心点)旋转一定角度的几何变换。在数学上,旋转属于刚体变换,可以保持图像中物体的形状和大小不变。

旋转的主要参数包括:

  • 旋转中心点

  • 旋转角度(顺时针或逆时针)

  • 旋转后的缩放比例

  • 旋转后图像的边界处理

3. 单点旋转与仿射变换矩阵

3.1 仿射变换基础

仿射变换是一种二维线性变换,可以表示为:

其中(x,y)是原坐标,(x',y')是变换后坐标。

在OpenCV中,我们使用2×3的矩阵来表示仿射变换:

 

3.2 获取旋转的仿射变换矩阵

OpenCV提供了cv2.getRotationMatrix2D函数来计算旋转矩阵:

import cv2
import numpy as np# 定义旋转中心和角度
center = (width/2, height/2)  # 通常以图像中心为旋转点
angle = 45  # 旋转角度
scale = 1.0  # 缩放比例# 获取旋转矩阵
rotation_matrix = cv2.getRotationMatrix2D(center, angle, scale)

3.3 应用仿射变换进行旋转

得到旋转矩阵后,可以使用cv2.warpAffine函数应用变换:

# 读取图像
image = cv2.imread('input.jpg')# 获取图像尺寸
height, width = image.shape[:2]# 应用旋转
rotated_image = cv2.warpAffine(image, rotation_matrix, (width, height))# 显示结果
cv2.imshow('Rotated Image', rotated_image)
cv2.waitKey(0)
cv2.destroyAllWindows()

4. 直接图片旋转

除了使用仿射变换,OpenCV还提供了更直接的旋转方式:

4.1 使用transpose和flip实现90度倍数的旋转

对于90°、180°、270°的旋转,可以使用更高效的操作:

# 顺时针旋转90度
rotated_90 = cv2.rotate(image, cv2.ROTATE_90_CLOCKWISE)# 逆时针旋转90度
rotated_90_counter = cv2.rotate(image, cv2.ROTATE_90_COUNTERCLOCKWISE)# 旋转180度
rotated_180 = cv2.rotate(image, cv2.ROTATE_180)

4.2 任意角度旋转的完整实现

对于任意角度的旋转,我们需要考虑旋转后图像可能超出原图边界的问题:

def rotate_image(image, angle):# 获取图像尺寸(h, w) = image.shape[:2]# 计算旋转中心center = (w // 2, h // 2)# 获取旋转矩阵M = cv2.getRotationMatrix2D(center, angle, 1.0)# 计算旋转后图像的边界cos = np.abs(M[0, 0])sin = np.abs(M[0, 1])new_w = int((h * sin) + (w * cos))new_h = int((h * cos) + (w * sin))# 调整旋转矩阵以考虑平移M[0, 2] += (new_w / 2) - center[0]M[1, 2] += (new_h / 2) - center[1]# 执行旋转rotated = cv2.warpAffine(image, M, (new_w, new_h), borderMode=cv2.BORDER_CONSTANT, borderValue=(255, 255, 255))return rotated

5. 旋转中的边界处理

旋转图像时,边角的处理非常重要。OpenCV提供了多种边界处理方式:

# 不同边界填充方式
rotated_replicate = cv2.warpAffine(image, M, (w, h), borderMode=cv2.BORDER_REPLICATE)
rotated_reflect = cv2.warpAffine(image, M, (w, h), borderMode=cv2.BORDER_REFLECT)
rotated_wrap = cv2.warpAffine(image, M, (w, h), borderMode=cv2.BORDER_WRAP)

6. 基于两点的旋转(扩展内容)

有时我们需要根据图像中的两个特征点来旋转图像,使两点连线达到特定角度:

def rotate_by_two_points(image, pt1, pt2, desired_angle=0):# 计算两点之间的角度dx = pt2[0] - pt1[0]dy = pt2[1] - pt1[1]current_angle = np.degrees(np.arctan2(dy, dx))# 计算需要旋转的角度rotation_angle = desired_angle - current_angle# 计算旋转中心(两点中点)center = ((pt1[0] + pt2[0]) / 2, (pt1[1] + pt2[1]) / 2)# 获取旋转矩阵M = cv2.getRotationMatrix2D(center, rotation_angle, 1.0)# 执行旋转rotated = cv2.warpAffine(image, M, (image.shape[1], image.shape[0]))return rotated

7. 性能优化与注意事项

  1. 批量旋转优化:如果需要旋转多张图片,可以预先计算旋转矩阵并复用

  2. 插值方法选择warpAffine中的flags参数可以指定插值方法,影响旋转质量

    • cv2.INTER_NEAREST:最近邻插值,速度快但质量低

    • cv2.INTER_LINEAR:双线性插值(默认)

    • cv2.INTER_CUBIC:双三次插值,质量更好但速度慢

  3. 内存考虑:大角度旋转会产生更大的图像,注意内存消耗

8. 实际应用案例

8.1 文档校正

def correct_document_skew(image):# 转换为灰度图gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)# 边缘检测edges = cv2.Canny(gray, 50, 150, apertureSize=3)# 霍夫线变换检测直线lines = cv2.HoughLinesP(edges, 1, np.pi/180, 100, minLineLength=100, maxLineGap=10)# 计算平均角度angles = []for line in lines:x1, y1, x2, y2 = line[0]angle = np.degrees(np.arctan2(y2-y1, x2-x1))angles.append(angle)median_angle = np.median(angles)# 旋转图像校正(h, w) = image.shape[:2]center = (w // 2, h // 2)M = cv2.getRotationMatrix2D(center, median_angle, 1.0)corrected = cv2.warpAffine(image, M, (w, h), flags=cv2.INTER_CUBIC,borderMode=cv2.BORDER_REPLICATE)return corrected

8.2 图像数据增强

在深度学习中,图像旋转是常用的数据增强手段:

def augment_data(image):# 随机旋转角度 (-15到15度之间)angle = np.random.uniform(-15, 15)# 随机缩放 (0.8到1.2之间)scale = np.random.uniform(0.8, 1.2)# 获取旋转矩阵h, w = image.shape[:2]center = (w/2, h/2)M = cv2.getRotationMatrix2D(center, angle, scale)# 应用变换augmented = cv2.warpAffine(image, M, (w, h), flags=cv2.INTER_LINEAR,borderMode=cv2.BORDER_REFLECT)return augmented

9. 总结

本文详细介绍了OpenCV中实现图像旋转的两种主要方法:基于单点的仿射变换旋转和直接图片旋转。关键点包括:

  1. 使用cv2.getRotationMatrix2D获取旋转矩阵

  2. 使用cv2.warpAffine应用仿射变换

  3. 处理旋转后的边界问题

  4. 基于两点旋转的特殊情况处理

  5. 实际应用中的性能优化技巧

掌握这些技术后,读者可以灵活地在各种计算机视觉应用中实现图像旋转功能。根据具体需求选择合适的方法,并注意旋转对图像质量的影响,就能获得最佳的旋转效果。

10. 参考文献

  1. OpenCV官方文档

  2. 《学习OpenCV》计算机视觉编程经典书籍

  3. 《数字图像处理》冈萨雷斯著

希望这篇文章能帮助你全面理解OpenCV中的图像旋转技术!如果有任何问题,欢迎在评论区留言讨论。

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

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

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

相关文章

yolov11安装,训练模型,tensorrtx加速,Qt预测图像

文章目录 一. yolov11 python环境安装二. windows10下yolov11 tensorrtx推理加速三. windows10下qt调用tensorrtx加速的yolov11进行检测 一. yolov11 python环境安装 基础环境 CUDA:cuda_11.8.0_522.06_windows cudnn:cudnn-windows-x86_64-8.6.0.163_c…

生物化学 PCR(聚合酶链式反应)引物 制造(固相磷酰胺化学法) 购买 存储

引物(Primer) 引物(Primer)是一小段单链 DNA(通常 18~25 个碱基),与模板 DNA 的特定位点互补。包括:Forward Primer(正向引物)和 Reverse Primer&#xff08…

SQL server 获取表中所有行的序号 不够四位数的前面补0

在 SQL Server 中,如果你想要为表中的行编号(即序号)添加前导零,以便它们总是呈现为四位数,你可以使用多种方法来实现这一点。以下是几种常用的方法: 方法1:使用 RIGHT 和 REPLICATE 函数 如果…

热门话题!网关模块解决AB机器人和电压控制器EtherCAT转Ethernet/IP难题

网关模块:解决AB机器人与电压控制器通讯难题 在现代工业自动化生产中,不同设备之间的通信与协同至关重要。然而,由于设备品牌、型号以及所采用的通信协议各异,常常会出现通信兼容性问题。本案例将详细介绍如何运用捷米特JM-ECTM-E…

将attribute数据动态写入到excel上

将attribute数据动态写入到excel上 显示效果: I 大体思路: excel range name就设置为attribute_数字_类型,在创建template的时候,通过API得到这个event有几个attribute,就创建几列,同时还要根据不同的类…

Stable Diffusion入门-ControlNet 深入理解 第一课:ControlNet,控制AI绘图的“大杀器”

大家好,欢迎来到Stable Diffusion入门-ControlNet深入理解系列的第一课! 今天,我们要聊聊一个让AI绘画从“盲目生成”走向“精准控制”的神奇插件——ControlNet。 它就像一位无声的魔术师,把原本随意的AI生成图片变得有条不紊、…

新生代潜力股刘小北:演艺路上的璀璨新星

在娱乐圈新人辈出的当下,一位来自四川的年轻演员正凭借着自己独特的魅力和扎实的演技,悄然走进观众的视野,他就是刘小北。1998年出生的刘小北,毕业于四川电影电视学院,自踏入演艺圈以来,便以坚定的步伐在演…

强制IDEA始终使用Java 8

解决IDEA总是使用Java 21而非Java 8编译的问题 您遇到的问题是典型的IDE内置JDK与项目冲突的情况。即使系统只安装了Java 8,IntelliJ IDEA仍内置有最新的Java运行时,导致它使用Java 21来编译您的代码。 解决方案:强制IDEA始终使用Java 8 1…

青少年编程与数学 01-012 通用应用软件简介 14 词典及翻译资源

青少年编程与数学 01-012 通用应用软件简介 14 词典及翻译资源 一、什么是词典及翻译资源(一)词典及翻译资源的基本定义(二)词典及翻译资源的工作原理(三)词典及翻译资源的类型 二、词典及翻译资源的重要意…

AI测试革命:5分钟自动生成单元测试|覆盖率和边界测试实战指南

AI测试革命:5分钟自动生成单元测试|覆盖率和边界测试实战指南 你是否曾为编写测试用例绞尽脑汁?是否因遗漏边界条件导致上线后BUG频发?告别低效测试,掌握AI赋能的现代化测试策略! 一、为什么我们需要AI测试…

n8n Docker Compose部署

n8n Docker Compose 部署官方文档详细总结 1. 前提条件 具备服务器、容器、网络和安全相关基础知识。推荐有 Linux 运维经验。已准备好一台服务器(建议为云服务器或本地服务器)。 2. 安装 Docker 和 Docker Compose 以 Ubuntu 为例,完整命…

Talk is cheap. Show me the code.手搓一个 Wayland 客户端程序

前几天我写了一篇万字长文《万字长文详解 Wayland 协议、架构》,但光讲协议分析难免有些枯燥。毕竟,程序员更信奉那句名言:Talk is cheap. Show me the code. 所以这篇文章不打算长篇大论,而是通过编写一个简单的 Wayland 客户端程…

Golang JSON 标准库用法详解

JSON (JavaScript Object Notation) 是一种轻量级的数据交换格式,Go语言的标准库encoding/json提供了强大的JSON处理能力。下面我将详细介绍各种用法并提供示例代码。 1. 基本编码(Marshal) 将Go数据结构转换为JSON字符串。 package maini…

Day.42

hook函数: import torch import torch.nn as nn import numpy as np import matplotlib.pyplot as plt torch.manual_seed(42) np.random.seed(42) 张量钩子: x torch.tensor([2.0], requires_gradTrue) y x ** 2 z y ** 3 def tensor_hook…

【.net core】【sqlsugar】在where条件查询时使用原生SQL

//初始化查询 var query repository.IQueryable();//添加原生SQL WHERE条件 query query.Where(" fieldA < 123"); 对应调用ISugarQueryable接口类中&#xff1a; ISugarQueryable<T> Where(string whereString, object parameters null);

网络 : 传输层【TCP协议】

网络 : 传输层【TCP协议】 一、TCP协议段格式1.1 32位序号与确认号1.1.1 32位序号1.1.2 确认号 1.2 4位首部长度1.3 6位标志位1.4 16位窗口大小 二、确认应答(ACK)机制三、超时重传机制四、连接管理机制4.1 三次握手(连接)listen的第二个参数 4.2 四次挥手(断开连接)**TIME_WAI…

人大金仓Kingbase数据库 Ksql: 未找到命令

人大金仓Kingbase数据库 Ksql: 未找到命令 1. 定位 Kingbase 安装目录 Kingbase 数据库通常安装在 /kingbase/ES/V8/Server 目录下。可以通过以下命令定位&#xff1a; cd /kingbase/ES/V8/Server2. 验证 ksql 工具是否安装成功 执行以下命令检查 ksql 客户端工具的版本信息…

Flask(四) 模板渲染render_template

文章目录 &#x1f4e6; 过程详解&#xff08;路由 <-> HTML 模板&#xff09;&#x1f9e0; 数据是怎么传过去的&#xff1f;多变量示例 ✅ Jinja2 支持条件判断、循环、模板继承&#xff1a;✅ 安全性&#x1f512; Flask 默认也会对变量进行 HTML 转义&#xff1a;&am…

[附源码+数据库+毕业论文+开题报告]基于Spring+MyBatis+MySQL+Maven+jsp实现的宠物领养管理系统,推荐!

摘 要 互联网发展至今&#xff0c;无论是其理论还是技术都已经成熟&#xff0c;而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播&#xff0c;搭配信息管理工具可以很好地为人们提供服务。针对宠物领养信息管理混乱&#xff0c;出错率高&#xff0c;信息安全性差…

【ArcGIS】水资源单项评价

【ArcGIS】水资源单项评价 一、水资源单项评价1、评价思路 二、操作步骤1、处理环境设置2、数据处理3、要素转栅格4、水资源评价 一、水资源单项评价 1、评价思路 &#xff08;1&#xff09;省级层面宜选用四级/五级水资源分区或县级行政区为评价单元&#xff0c;按照水资源总…