(一) 图像增广
对图像进行随机变化(翻转和裁剪)能够扩大训练集的规模也能减少模型对某些属性的依赖,提高泛化能力.
torchvision.transforms
模块提供了翻转和裁剪的方法
RandomHorizontalFlip()
50%几率使得图像向左或向右翻转
RandomVerticalFlip()
50%几率使得图像向上或向下翻转
RandomResizedCrop((200, 200), scale=(0.1, 1), ratio=(0.5, 2))
随机裁剪原始面积0.1到1的区域,宽高比在0.5-2,宽度和高度缩放到200像素
ColorJitter(brightness=0.5, contrast=0, saturation=0, hue=0)
调整亮度、对比度、饱和度和色调. 0.5就是0-0.5
Compose()
提供了将多个增广方法数组合并为一个方法的方法
(二) 微调
有些模型需要大量的数据集进行训练(比如Fashion-MNIST图像数据集),如果实例数据不够,那么很可能过拟合.解决办法除了收集更多的数据,还可以"迁移学习"将源数据集的知识迁移到目标数据集.
步骤:
- 在源数据集上预训练神经网络模型(源模型)
- 创建一个新的神经网络模型(目标模型),复制源模型上的所有模型设计及其参数(输出层除外)
- 向目标模型添加输出层,其输出数是目标数据集中的类别数.然后随机初始化该层的模型参数.
- 训练目标模型.输出层从头训练但是其他层都是只是微调.
(三) 目标检测和边界框
目标检测就是图像中有多个目标我们既想知道他们的类别又想知道他们在图像中的具体位置.而边界框就是确定目标位置的一种表示方法.
边界框顾名思义就是一个矩形,可以由左上角和右下角两点确定,也可以由中心点和宽高决定.如果要将边界框在图中画出来,可以将边界框表示为matplotlib
的边界框格式然后再axes.add_patch()
即可画出.
(四) 锚框
目标检测算法会在输入图像中采样大量区域然后判断.以每个像素为中心生成多个缩放比和宽高比不同的边界框,这些框就是锚框.
在实践中,我们用的是缩放比s和宽高比r.选择一些缩放比s1,...,sn和宽高比r1,...,sm,由于复杂度我们可以只考虑这些锚框(共n+m-1个):
(s1,r1),(s1,r2),...,(s1,r2),(s1,rm),(s2,r1),...,(sn,r1)
生成锚框坐标张量的实例代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
| def multibox_prior(data, sizes, ratios): """生成以每个像素为中心具有不同形状的锚框""" in_height, in_width = data.shape[-2:] device, num_sizes, num_ratios = data.device, len(sizes), len(ratios) boxes_per_pixel = (num_sizes + num_ratios - 1) size_tensor = torch.tensor(sizes, device=device) ratio_tensor = torch.tensor(ratios, device=device)
offset_h, offset_w = 0.5, 0.5 steps_h = 1.0 / in_height steps_w = 1.0 / in_width
center_h = (torch.arange(in_height, device=device) + offset_h) * steps_h center_w = (torch.arange(in_width, device=device) + offset_w) * steps_w shift_y, shift_x = torch.meshgrid(center_h, center_w, indexing='ij') shift_y, shift_x = shift_y.reshape(-1), shift_x.reshape(-1)
w = torch.cat((size_tensor * torch.sqrt(ratio_tensor[0]), sizes[0] * torch.sqrt(ratio_tensor[1:])))\ * in_height / in_width h = torch.cat((size_tensor / torch.sqrt(ratio_tensor[0]), sizes[0] / torch.sqrt(ratio_tensor[1:]))) anchor_manipulations = torch.stack((-w, -h, w, h)).T.repeat( in_height * in_width, 1) / 2
out_grid = torch.stack([shift_x, shift_y, shift_x, shift_y], dim=1).repeat_interleave(boxes_per_pixel, dim=0) output = out_grid + anchor_manipulations return output.unsqueeze(0)
|
只是做一个参考,我们完全可以自己实现.
交并比(intersection over union,loU)
如果需要描述锚框和真实边界框之间的相似性,可以用杰卡德系数表示(两个边界框的杰卡德系数就叫交并比),一张图就很显然了:

锚框与真实边界框匹配
原文书上很长一段话,其实算法很简单,列一个交并比矩阵(其实就是列表):
|
B1 |
B2 |
… |
Bnb |
A1 |
J11 |
J12 |
|
J1nb |
A2 |
J21 |
J22 |
|
J2nb |
… |
|
|
|
|
Ana |
Jna1 |
Jna2 |
|
Jnanb |
我们要做的就是对于每一个锚框分配一个最接近的真实边界框,那么显然我们可以先找这个矩阵的所有元素中大的那个元素(也就是匹配的最好的),然后找到之后就把这一列和这一行划掉(代表这个A已经和这个B匹配了),当所有B都被分配完了还剩下一些A没有分配怎么办呢?那就只能将剩下的这些A(也就是这些行中)找到最佳匹配的B,而且需要大于预定义的阈值的时候才把B分配给这些剩下的A.
不过我有一个疑问:为什么不直接每个A(也就是每一行)直接找最匹配的B?是为了尽可能避免重复吗?
偏移量
当一个锚框分配一个真实边界框的时候也就是这个锚框类别被标记和这个真实边界框的类别相同了.如果锚框没有被分配真实边界框(因为可能低于预定义的阈值),将其标记为background(背景、负类锚框)
中心坐标分别为(xa,ya)(xb,yb)两个框的偏移(相差)可以用以下式子表示:
(σxwaxb−xa−μx,σyhayb−ya−μy,σwlogwawb−μw,σhloghahb−μh)
常量的默认值为μx=μy=μw=μh=0,σx=σy=0.1,σw=σh=0.2.负类锚框的偏移量被标记为0.
偏移量的四个维度有什么含义?
预测
将锚框和预测偏移量作为输入,用逆偏移变换返回预测的边界框坐标.当有许多锚框时,可能会输出许多相似的具有明显重叠的预测边界框,都围绕着同一目标.为了简化输出,我们可以使用非极大值抑制(non-maximum suppression,NMS)合并属于同一目标的类似的预测边界框.非极大值抑制就是不断将置信度最高的预测边界框B1作为基准,然后将所有与B1的IoU超过预定阈值ϵ的非基准预测边界框移除.
代码暂时没有贴…