第四篇 FastAI中的数据增强

2024-02-27 00:32
文章标签 数据 增强 第四篇 fastai

本文主要是介绍第四篇 FastAI中的数据增强,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一篇博客介绍了如何使用Fast AI数据模块(Data Block),便捷地构建Fast AI模型所需的数据包(Data Bunch)。在将图像数据灌入模型之前,往往需要对之进行随机变换,即做数据增强(Data Augmentation)。这可以视为一种在数据层面的正则化(也就是人为地引入一些随机扰动,避免学习器过分关注训练集的特有性质,以免产生过拟合)。本篇博客将介绍Fast AI中的数据增强模块,相关文档见文档链接。

一、Transform基类

Fast AI提供了许多图像变换函数(会在本博客的后面提到),使用这些变换函数,可以直接对图像进行变换。同时,为了给这些变换函数提供数据增强时的随机机制,Fast AI提供了一个封装类—Transform类(定义在fastai/vision/image.py文件中),该类的功能主要还包括对变换函数添加一些共有特性:如变换函数的优先级(order)、包装器的名称(变量为_wrap,对应于Image类中的相应函数,用于对变换函数调用之后的结果做进一步的操作)等。当使用该类封装的变换函数时,如果按照正常的变换函数传入相关参数,则效果与直接调用该变换函数一致,此时没有随机性因素。以Fast AI库提供的用于改变图像明暗的brightness类为例:

brightness_func = brightness(change=0.2, p=0.5)
img = open_image(img_file)
y = brightness_func(img, change=0.2)

注意:在使用Transform类包装的变换函数时直接作用于图像数据时,要按照原变换函数的签名提供完整的参数列表,如上例中change=0.2

如果使用img.apply_tfms()方式调用,则此时会存在两种随机机制:当被标注为uniform的参数为固定值时,则按照提供的p值,以概率p进行变换;当被标注为uniform的参数为区间时,则变换以概率p发生,并且该参数在所提供的区间内随机选择。

二、变换函数

相应代码见fastai/vision/transform.py

函数签名说明
brightness(x, change: uniform) → Image :: TfmLighting改变图像明暗,通过对图像的logit pixel进行加减常量实现。当change=0.5时,图像无变化;当change=1时,图像会变换为白色;当change=0时,图像会变换为白色。
contrast(x, scale:log_uniform) → Image :: TfmLighting调整对比度,通过对图像的logit pixel乘上一个常量实现。 当scale=0时,会将图像转换为灰色,当scale>1会增强图像的对比度(即明暗像素的差异更大);当scale=1时,不调整对比度。
crop(x, size, row_pct:uniform=0.5, col_pct:uniform=0.5) → Image :: TfmPixel图像裁剪,其中(row_pct, col_pct)限定了裁剪框锚点的位置,以归一化的比例进行表示。
crop_pad(x, size, padding_mode='reflection', row_pct:uniform=0.5, col_pct:uniform=0.5) → Image :: TfmCrop类似于crop(),不过裁剪框的大小可以超出图像范围,填充方法通过padding_mode指定,可选为reflectionzerosborder
dihedral(x, k:partial(uniform_int, 0, 7)) → Image :: TfmPixel镜像翻转与旋转90°。
flip_lr(x) → Image :: TfmPixel水平翻转。
jitter(c, magnitude:uniform) → Image :: TfmCoord邻域像素替换, 邻域范围由magnitude限定。
perspective_warp(c, magnitude:partial(uniform, size=8)=0, invert=False) → Image :: TfmCoord透视变换,其中manigtude为8元素集,指定了将四个角的归一化坐标变换的幅度。默认填充方法是reflection
Image.resize(self, size:Union[int,TensorImageSize])->'Image'图像缩放,使用的是torch中相应的方法。对图像而言,size为一个整数,或者TensorImageSize类型·(3, H, W),默认使用SQUISH方法。在使用数据模块的API构建数据包时,可通过设置resize_method来选择处理方式。
rotate(degrees:uniform) → Image :: TfmAffine图像旋转。
rgb_randomize(x, channel:int=None, thresh:float=0.3) → Image :: TfmPixel随机化RGB的某一通道,通过thresh限定该通道的最大值。
skew(c, direction:uniform_int, magnitude:uniform=0, invert=False) → Image :: TfmCoord扭曲,实际是通过perspective_warp()实现的。
squish(scale:uniform=1.0, row_pct:uniform=0.5, col_pct:uniform=0.5) → Image :: TfmAffine拉伸,scale<1时,为横向拉伸;scale>1时,为纵向拉伸。
symmetric_warp(c, magnitude:partial(uniform, size=4)=0, invert=False) → Image :: TfmCoord特定的透视变换。
tilt(c, direction:uniform_int, magnitude:uniform=0, invert=False) → Image :: TfmCoord倾斜。
zoom(scale:uniform=1.0, row_pct:uniform=0.5, col_pct:uniform=0.5) → Image :: TfmAffine等比例缩放。
cutout(x, n_holes:uniform_int=1, length:uniform_int=40) → Image :: TfmPixel制造孔洞。

三、get_transforms()函数

该函数会返回变换函数的两个列表,一个用于训练集,一个用于验证集。

get_transforms(do_flip:bool=True, # 是否进行水平翻转flip_vert:bool=False, #是否进行垂直翻转    max_rotate:float=10.0, max_zoom:float=1.1, max_lighting:float=0.2, max_warp:float=0.2, p_affine:float=0.75, p_lighting:float=0.75, xtra_tfms:Optional[Collection[Transform]]=None) → Collection[Transform]

具体而言,在构建数据包时,按如下方式进行使用:

data = ImageDataBunch.from_folder(path, ds_tfms=tfms, size=26)

或者

tfms = get_transforms(flip_vert=True, max_lighting=0.1, max_zoom=1.05, max_warp=0.)
data = (ImageList.from_folder(path) # 数据文件的路径 .split_by_folder()      # 按比例分割训练集和验证集    .label_from_folder()    # 指定类别标签    .transform(tfms, size=32)     # 对图像进行变换    .databunch(bs=128)    .normalize(imagenet_stats) # 数据归一化)

四、一些补充

1. fastai.vision.Image类中的图像数据究竟是以pixel存储的还是以logit存储的?

只能说,当你需要它是pixel时,它就是pixel,即img.px;当你需要它是logit_pixel时,它就是logit_pixel,即img.logit_px。这是通过Image类中的refresh()函数实现的,而每次访问img.data时,总会调用refresh()函数。如果检查到img.logit_px不为None,则可认为logit_px是最新的操作结果,则通过sigmoid函数将其变换为img.px,并置img.logit_px=None;否则返回存储的img.px。另外,需要访问img.logit_px时,若img.logit_px is None,则计算img.pxlogit,并存储在img.logit_px中。

2. crop()函数中的锚点位置图示

见下图,这样做的好处是无需判断裁剪框是否超出图像边界。

图 1. crop函数中裁剪框锚点图示
3. dihedral()函数结果图示
图 2. 翻转函数效果图示
4. 对Image对象的像素坐标的更改

jitter()这类的标记为TfmCoord的变换函数,会对图像像素坐标进行变换。这一过程是这样完成的:首先生成图像对象每个像素的归一化坐标网格(坐标分布在[-1, 1]),这一值会存储在img.flow属性中;然后对该网格进行变换;在图像像素值获取时,做完logit_pxpx的转换后,就依据img.flow对图像进行重采样。这些都是在Image.refresh()函数中实现的。

5. 补全为什么总是reflection

TfmCoordTfmAffine类的变换函数,会涉及图像像素的坐标变换,而这些变换都是通过_grid_sample()函数完成的,而_grid_sample()函数的默认补全方法就是reflection

6. Fast AI的并行处理函数函

数为fastai/core.py里的parrallel,需要传入要执行的函数和参数列表。示例如下(文档链接):

num_cpus() #获取CPU的数目
def my_func(value, index):print("Index: {}, Value: {}".format(index, value)) my_array = [i*2 for i in range(5)]
parallel(my_func, my_array, max_workers=3)

这篇关于第四篇 FastAI中的数据增强的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/750667

相关文章

MySQL快速复制一张表的四种核心方法(包括表结构和数据)

《MySQL快速复制一张表的四种核心方法(包括表结构和数据)》本文详细介绍了四种复制MySQL表(结构+数据)的方法,并对每种方法进行了对比分析,适用于不同场景和数据量的复制需求,特别是针对超大表(1... 目录一、mysql 复制表(结构+数据)的 4 种核心方法(面试结构化回答)方法 1:CREATE

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

MySQL中的DELETE删除数据及注意事项

《MySQL中的DELETE删除数据及注意事项》MySQL的DELETE语句是数据库操作中不可或缺的一部分,通过合理使用索引、批量删除、避免全表删除、使用TRUNCATE、使用ORDERBY和LIMI... 目录1. 基本语法单表删除2. 高级用法使用子查询删除删除多表3. 性能优化策略使用索引批量删除避免

MySQL 数据库进阶之SQL 数据操作与子查询操作大全

《MySQL数据库进阶之SQL数据操作与子查询操作大全》本文详细介绍了SQL中的子查询、数据添加(INSERT)、数据修改(UPDATE)和数据删除(DELETE、TRUNCATE、DROP)操作... 目录一、子查询:嵌套在查询中的查询1.1 子查询的基本语法1.2 子查询的实战示例二、数据添加:INSE

Linux服务器数据盘移除并重新挂载的全过程

《Linux服务器数据盘移除并重新挂载的全过程》:本文主要介绍在Linux服务器上移除并重新挂载数据盘的整个过程,分为三大步:卸载文件系统、分离磁盘和重新挂载,每一步都有详细的步骤和注意事项,确保... 目录引言第一步:卸载文件系统第二步:分离磁盘第三步:重新挂载引言在 linux 服务器上移除并重新挂p

使用MyBatis TypeHandler实现数据加密与解密的具体方案

《使用MyBatisTypeHandler实现数据加密与解密的具体方案》在我们日常的开发工作中,经常会遇到一些敏感数据需要存储,比如用户的手机号、身份证号、银行卡号等,为了保障数据安全,我们通常会对... 目录1. 核心概念:什么是 TypeHandler?2. 实战场景3. 代码实现步骤步骤 1:定义 E

使用C#导出Excel数据并保存多种格式的完整示例

《使用C#导出Excel数据并保存多种格式的完整示例》在现代企业信息化管理中,Excel已经成为最常用的数据存储和分析工具,从员工信息表、销售数据报表到财务分析表,几乎所有部门都离不开Excel,本文... 目录引言1. 安装 Spire.XLS2. 创建工作簿和填充数据3. 保存为不同格式4. 效果展示5

Python多任务爬虫实现爬取图片和GDP数据

《Python多任务爬虫实现爬取图片和GDP数据》本文主要介绍了基于FastAPI开发Web站点的方法,包括搭建Web服务器、处理图片资源、实现多任务爬虫和数据可视化,同时,还简要介绍了Python爬... 目录一. 基于FastAPI之Web站点开发1. 基于FastAPI搭建Web服务器2. Web服务

MySQL 批量插入的原理和实战方法(快速提升大数据导入效率)

《MySQL批量插入的原理和实战方法(快速提升大数据导入效率)》在日常开发中,我们经常需要将大量数据批量插入到MySQL数据库中,本文将介绍批量插入的原理、实现方法,并结合Python和PyMySQ... 目录一、批量插入的优势二、mysql 表的创建示例三、python 实现批量插入1. 安装 PyMyS

关于MySQL将表中数据删除后多久空间会被释放出来

《关于MySQL将表中数据删除后多久空间会被释放出来》MySQL删除数据后,空间不会立即释放给操作系统,而是会被标记为“可重用”,以供未来插入新数据时使用,只有满足特定条件时,空间才可能真正返还给操作... 目录一、mysql数据删除与空间管理1.1 理解MySQL数据删除原理1.3 执行SQL1.3 使用