图像基本变换

news/2024/4/29 6:11:09/文章来源:https://blog.csdn.net/qq_54809548/article/details/129655600

缩放与裁剪

裁剪

图像的裁剪,是指将图像的某个区域切割出来。

一些常见的应用场景包括:

* 感兴趣区域提取

* 去除无用信息

* 图像增强

* 纠偏:去除不规则部分,将图像变得更加整齐

事实上,图像裁剪的裁剪通常就是一个numpy矩阵切片的过程。

其中的关键内容在于,获取目标区域的坐标。

给定一个512*512的lena图像,想要裁剪出以原图中心点为中心的,320*320的图像,代码如下所示:

import cv2
import matplotlib.pyplot as pltimg = cv2.imread('./dataset/lena.png')
img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
plt.imshow(img)
plt.axis('off')
plt.show()
# numpy切片
h, w = img.shape[:2]
cropped_img = img[h // 2 - 160: h // 2 + 160, w // 2 - 160: w // 2 + 160, :]
plt.imshow(cropped_img)
plt.show()#PIL库
from PIL import Image
cropped_img = img.crop((512 // 2 - 320 // 2, 512 // 2 - 320 // 2,  # 左上角坐标512 // 2 + 320 // 2, 512 // 2 + 320 // 2)) # 右下角坐标
plt.imshow(cropped_img)
plt.show()

缩放

cv2.resize(img, dsize=(, ), interpolation=cv2.INTER_AREA)

在resize函数的参数中,第一个参数为待缩放图片,第二个参数为缩放后的尺寸。

第三个参数为插值方法,共包括:

* cv2.INTER_NEAREST:最近邻插值。

* cv2.INTER_LINEAR:双线性插值。

* cv2.INTER_CUBIC:双立方插值。

* cv2.INTER_AREA:区域插值。

* cv2.INTER_LANCZOS4:Lanczos插值

以上插值方法特点不同,下面展示其中区别

img = cv2.imread('./dataset/lena.png')
img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
resized_img_1 = cv2.resize(img, dsize=(256, 384), interpolation=cv2.INTER_AREA)
plt.imshow(resized_img_1)
plt.show()

插值

* 最近邻插值: 源图像中距离新像素位置最近的像素赋值给新像素。

* 例如,图像缩放了fx和fy倍,那么新图像中的像素坐标$x', y'$对应原图坐标为$f_xx, f_yy$

* 优点:速度快

* 缺点:未考虑周围像素,锯齿严重

* 双线性插值:从周围四个像素共同加权计算像素值,距离越近权重越大,距离越远权重越小

* 优点:平滑

* 速度慢

大部分使用

* 双立方插值:新图像的每个点,都参考了周围16个点的信息

* 两个基本步骤:

- 计算周围16个点的像素值

- 拟合函数计算最终像素值

* 优点:保留图像细节信息

* 缺点:计算量大,耗时长

* 区域重采样:按区域计算新像素

* 每个像素都选取一个固定大小的区域

* 每个区域通过一定算法计算一个新的像素值

* 优点:保真,不易出现伪影

* lanczos:造了一个核函数,计算新像素

* 优点:更加平滑的采样结果

* 缺点:计算成本高

其中,a为核半径,在目标像素的a范围内,通过公式计算新的像素值。

target_img = img[240: 272, 240: 272, :]
# plt.imshow(target_img)
# plt.show()nearest = cv2.resize(target_img, (768, 768), interpolation=cv2.INTER_NEAREST)
linear = cv2.resize(target_img, (768, 768), interpolation=cv2.INTER_LINEAR)
cubic = cv2.resize(target_img, (768, 768), interpolation=cv2.INTER_CUBIC)
area = cv2.resize(target_img, (768, 768), interpolation=cv2.INTER_AREA)
lanczos = cv2.resize(target_img, (768, 768), interpolation=cv2.INTER_LANCZOS4)
fig, ax = plt.subplots(nrows=1, ncols=5, figsize=(12,12))
print(ax)
ax[0].imshow(nearest)
ax[1].imshow(linear)
ax[2].imshow(cubic)
ax[3].imshow(area)
ax[4].imshow(lanczos)ax[0].axis('off')
ax[1].axis('off')
ax[2].axis('off')
ax[3].axis('off')
ax[4].axis('off')plt.show()

仿射变换和透视变换

1. 仿射变换

https://blog.csdn.net/weixin_51571728/article/details/124434728

对于二维平面来说,仿射变换就是将一个点,通过**线性变换**映射到另一个二维平面中的点。

平行原则:原来是平行线,仿射后也是平行线

直线原则:原来是直线,仿射以后也是直线

与之相对的,**非线形变换**不是仿射变换,因为其不满足仿射变换的线性性质。

一个典型的非仿射变换为透视变换

对于矩阵来说,仿射变换的结果,可以表示为矩阵的乘积

仿射变换通常包含以下几种操作

* 平移:沿着x,y方向移动图像

* 旋转:围绕中心点旋转图像

* 缩放:缩放到指定大小

* 错切:在平面上将图像斜切一定角度

仿射变换可用于图像对齐、图像矫正、图像增强等。

图像旋转

图像旋转是一种特殊的仿射变换

PIL中,图像旋转有专用的函数接口img.rotate(角度,center=[ , ]),示例如下:

import PIL.Image as Image
import matplotlib.pyplot as pltimg = Image.open('E:/notebook/3c.png')
# 绕中心点逆时针转45度
rotated_img = img.rotate(45, center=[img.width // 2, img.height // 2])
fig, (ax1, ax2) = plt.subplots(nrows=1, ncols=2)
ax1.imshow(img)
ax2.imshow(rotated_img)
ax1.axis('off')
ax2.axis('off')
plt.show()

OpenCV提供了更加便捷灵活的函数方法:

首先,通过指定的中心和旋转角度,定义旋转矩阵:cv2.getRotationMatrix2D((旋转中心), 角度, 缩放比例)

其次,根据旋转矩阵,进行仿射变换,实现旋转:cv2.warpAffine(img, 旋转矩阵, (运算后矩阵的大小也就是输出图片的尺寸) )https://blog.csdn.net/m0_51545690/article/details/123959995

import cv2
img = cv2.imread('E:/notebook/3c.png')
img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
M = cv2.getRotationMatrix2D((img.shape[0] // 2, img.shape[1] // 2), 45, 1)
# print(M) # 旋转矩阵
#[[   0.70710678    0.70710678 -102.64170235]
# [  -0.70710678    0.70710678  208.20101013]]
# # 得到矩阵后得用到图像的仿射变换函数才可以进行最终图像的变化
rotate_img = cv2.warpAffine(img, M, (512,512))
fig, (ax1, ax2) = plt.subplots(nrows=1, ncols=2)
ax1.imshow(img)
ax2.imshow(rotate_img)
ax1.axis('off')
ax2.axis('off')
plt.show()

不依赖第三方库,将一个图像按照中心点,缩小为原面积的四分之一。

import cv2
img = cv2.imread('E:/notebook/lena.png')
img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
def resize(image):# image是读取用opencv读取完毕的numpy矩阵# 输出为仿射变换后的矩阵,注意要将其转化为适合图像的数据类型# 此处撰写代码M = np.array([[.5, 0, 0],[0, .5, 0]])# 首先,将图像的平面坐标扩展出一个1,可直接相乘img_matrix = np.array([[i, j, 1] for j in range(512) for i in range(512)])img_matrix = img_matrix.T# 其次,乘以矩阵得到新的坐标rotated_img_matrix = np.dot(M, img_matrix)rotated_img_matrix = np.rint(rotated_img_matrix).astype(int)# 最后,按照新坐标填补像素resized_image = np.zeros_like(img)for i in range(512):for j in range(512):num = i * 512 + jx = rotated_img_matrix[1, num]y = rotated_img_matrix[0, num]if 0 < x < 512 and 0 < y < 512:resized_image[x, y] = img[i, j]return resized_image
final_img = resize(img)
plt.imshow(final_img)
plt.show()

2. 透视变换

原理:https://blog.csdn.net/m0_43609475/article/details/112847314

区别于仿射变换,透视变换是通过变换矩阵对图像进行变形,来实现视角的转换。

在透视变换中,不仅会发生平移或者旋转,也会发生图像形变。

仿射变换需要三点求解仿射矩阵,而透视变换则需要四点标定,来获得变换矩阵

常见应用:鱼眼畸变校正、鸟瞰视角变换

opencv函数

getPerspectiveTransform([左上],[左下],[右上],[右下]四个坐标)

warpPerspective(img, 旋转矩阵, 输出图片大小)

例:将图像变为正对镜头的视角

import cv2img = cv2.imread('E:/notebook/grid.png')
img = cv2.cvtColor(img, cv2.COLOR_BGR2RGB)
print(img.shape)
M = cv2.getPerspectiveTransform(np.float32([[410, 392], [506, 677], [828, 283], [897, 635]]), np.float32([[0, 0], [0, 980], [1272, 0], [1272, 980]]))
print(M)
plt.imshow(img)
plt.show()
project_img = cv2.warpPerspective(img, M, (1272, 980))
plt.imshow(project_img)
plt.show()

https://blog.csdn.net/m0_51653200/article/details/127361624

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

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

相关文章

湖科大嵌入式实验课配套的stem32开发板使用说明

前言 本文档为湖科大嵌入式实验课配套的stem32开发板使用说明 这门课叫嵌入式系统及应用 大三下开课 板子长这样 感谢 lcw 的指导&#xff0c;让我少走弯路 安装 打开百度网盘链接 下载并安装keil 5 (MDK535.EXE) mdk518.exe应该也可以用&#xff0c;而且这个也可以汉化…

2022年18个值得期待的Learndash LMS插件列表

有数百个独特的LearnDash附加组件&#xff0c;您可能很难选择您的LearnDash LMS所需的。您可以集成 LearnDash 以扩展或限制功能&#xff0c;但您喜欢它。但真正的问题是如何为您的网站选择合适的插件&#xff1f;为了帮助简化此过程&#xff0c;我们精选了18个最值得期待的Lea…

git 分支创建、切换和合并

1.理解分支为了便于理解&#xff0c;大家可以粗略的将分支认为就是一个代码的副本。如果我们同时在一个代码上开发多个功能。还要修改一些bug&#xff0c;团队成员协作过程中&#xff0c;必然会出现相互影响。假如某个同事提交了一个错误的代码&#xff0c;可能会导致其他更新了…

设计循环队列(LeetCode)题目

这个也是力扣的题目&#xff0c;所以我们还是直接看 请自己看题目 下面就看思路吧&#xff0c;首先是循环队列&#xff0c;我们想一下循环队列如何设计&#xff0c;用什么设计比较好一点&#xff0c;用顺序表还是链表 这里用链表看起来比较简单&#xff0c;因为他是循环的&am…

干货|小白1分钟搞懂SRM管理系统

SRM管理系统是用来管理供应商&#xff0c;以及采购方和供应商进行采购交易的一套管理系统&#xff0c;在很多生产类企业中&#xff0c;往往是需要和ERP管理系统集成使用。原因&#xff1a;1.ERP管理系统无法实现采购商与供货商的有效协作&#xff1b;2.SRM管理系统对供应商的管…

Java分布式事务(七)

文章目录&#x1f525;Seata提供XA模式实现分布式事务_业务说明&#x1f525;Seata提供XA模式实现分布式事务_下载启动Seata服务&#x1f525;Seata提供XA模式实现分布式事务_搭建聚合父工程构建&#x1f525;Seata提供XA模式实现分布式事务_转账功能实现上&#x1f525;Seata提…

考研复试7 汇编语言、编程语言

一、寄存器 1. 寄存器概述 &#xff08;1&#xff09;典型的CPU包括器件 运算器控制器总线&#xff1a;内部总线实现CPU内部各个器件之间的联系&#xff1b;外部总线实现CPU和主板上其它器件的联系。 &#xff08;2&#xff09;8086CPU有14个寄存器&#xff0c;它们的名称为…

git查漏补缺之 stash

git查漏补缺之 stash Start 最近在工作的过程中&#xff0c;遇到 git 中的stash 暂存这个命令&#xff0c;感觉非常有用&#xff0c;写一个博客记录一下。 1. stash 首先第一个就是 stash &#xff0c;英译过来的意思就是 存放/贮藏。 主要的使用场景&#xff1a; 我正在…

最好用的Markdown编辑器:MWeb Pro mac

MWeb pro Mac中文版是一款非常好用的Markdown编辑器和博客生成工具&#xff0c;支持语法高亮&#xff0c;预览&#xff0c;Fenced code blocks和代码高亮&#xff0c;Math ML支持&#xff0c;具有导出HTML/PDF&#xff0c;自定编辑器主题&#xff0c;字数统计&#xff0c;大纲视…

Linux--MySQL数据库的基本概念、编译安装以及MySQL数据库自动补全功能的实现

文章目录一.数据库的基本概念1、数据&#xff08;Data&#xff09;2、表3、数据库4、数据库管理系统&#xff08;DBMS&#xff09;5、数据库系统6、访问数据库的流程二.数据库系统发展史1、第一代数据库2、第二代数据库3、第三代数据库三、当今主流数据库介绍1、SQL Server (微…

电商出海:阿里、拼多多“快马扬鞭”

配图来自Canva可画 经过多年的发展&#xff0c;电商已经逐渐深入人们的日常生活中了&#xff0c;人们也愈发习惯使用电商平台了。得益于消费者需求的持续增长&#xff0c;电商行业的体量规模也在持续扩大&#xff0c;行业内也涌现出了诸多电商平台&#xff0c;比如淘宝、京东、…

error: C1083: 无法打开包括文件: “QtGui/QApplication”: No such file or directory

Qt系列文章目录 文章目录Qt系列文章目录前言一、原因二、解决办法1.修改pro工程文件2.在main.cpp中三、总结前言 当我们从网上或者从打开别人的工程师&#xff0c;报错&#xff0c;C1083: 无法打开包括文件: “QtGui/QApplication”。 原因&#xff1a;Qt5里不再用QtGui模块&a…

docker 安装 nginx无坑版

一. 拉取镜像 docker pull nginx二. 创建挂载目录 mkdir -p /usr/local/nginx/conf mkdir -p /usr/local/nginx/log mkdir -p /usr/local/nginx/html三. 从nginx容器里复制nginx的配置文件到主机里 创建个容器 docker run --name nginx -p 80:80 -d nginx将容器内的配置文件…

mysql之窗口函数练习

&#x1f34a;今天复习一下mysql中的窗口函数&#xff0c;主要是通过几道练习题复习和加深一下对窗口函数的理解&#xff0c;对往期内容感兴趣的同学可以参考如下内容&#x1f447;: 链接: 牛客SQL大厂真题——某音短视频.链接: 京东数据分析SQL面试题.链接: 百度用户增长SQL面…

Java实习生------MySQL10道面试题打卡

今日语录&#xff1a;“没有执行力&#xff0c;就没有竞争力 ”&#x1f339; 参考资料&#xff1a;图解MySQL、MySQL面试题 1、事务有哪些特性&#xff1f; 原子性&#xff1a; 一个事务中的所有操作&#xff0c;要么全部完成&#xff0c;要么全部不完成&#xff0c;不会出现…

Linux系统的安装以及参数配置 -- VMware(虚拟机)安装--ubuntu 20.04--VMware Tools工具安装

Linux系统的安装以及参数配置 PS&#xff1a;本文章为上课后整理笔记&#xff0c;作为以后学习工作的学习使用&#xff0c;也作为一次课程记录 一、Linux系统的安装常用方法 – 3种 1.直接Linux操作替换Windows – 专业Linux开发者 – 接受Linux相关软件的使用 2.安装双系统…

诗佛王维,眼前的苟且和远方的田野?

转自&#xff1a;媲美李白杜甫的诗人&#xff0c;他的人生可以复制_百科TA说 (baidu.com)他受到的羁绊&#xff0c;他做出的选择&#xff0c;提供了一种温润平和的过日子模式。大部分人无法决绝地脱离社会&#xff0c;隐遁起来&#xff0c;也无法在社会中不计底线&#xff0c;混…

JavaScript性能优化小窍门汇总(含实例)

在众多语言中&#xff0c;JavaScript已经占有重要的一席之地&#xff0c;利用JavaScript我们可以做很多事情 &#xff0c; 应用广泛。在web应用项目中&#xff0c;需要大量JavaScript的代码&#xff0c;将来也会越来越多。但是由于JavaScript是一个作为解释执行的语言&#xff…

Vue|样式绑定

class 与 style 是 HTML 元素的属性&#xff0c;用于设置元素的样式&#xff0c;我们可以用 v-bind 来设置样式属性。Vue.js v-bind 在处理 class 和 style 时&#xff0c; 专门增强了它。表达式的结果类型除了字符串之外&#xff0c;还可以是对象或数组。 文末名片获取源码 精…

根据平均分来划分等级-课后程序(JavaScript前端开发案例教程-黑马程序员编著-第2章-课后作业)

【案例2-1】 根据平均分来划分等级 一、案例描述 考核知识点 switch语句 练习目标 掌握switch语句的使用。 需求分析 switch语句也是多分支语句&#xff0c;针对某个表达式的值做出判断&#xff0c;来决定执行哪一段代码&#xff0c;本案例用于实现根据输入的小明同学的5门课…