IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 人工智能 -> YOLO-V3-SPP utils.py build_targets函数-详细解读(ultralytic版本) -> 正文阅读

[人工智能]YOLO-V3-SPP utils.py build_targets函数-详细解读(ultralytic版本)

前言

理论详解:YOLO-V3-SPP详细解析

build_targets

讲解形式主要以流程图形式,逐流程详细解读每一行代码

代码以pytorch框架为基础

targets处理整体流程

这里主要介绍了targets的来龙去脉,targets指的是数据集中标注好的GroundTruth的目标信息,,build_target这个函数主要是处理当前批次的所有图片的targets,将当前批次的所有targets经过:

  1. 宽高IOU筛选
  2. 标注的yolo格式的box信息 ( x c e n t e r , y c e n t e r , w , h ) (x_{center},y_{center},w,h) (xcenter?,ycenter?,w,h)转化为 ( t x , t y , w , h ) (t_x,t_y,w,h) (tx?,ty?,w,h),其中 t x t_x tx? t y t_y ty?分别表示当前box中心离所在grid_cell左上角坐标的偏移量

博文主要讲解targets的筛选过程,主要讲清楚build_targets这个函数在做什么

请添加图片描述

build_targets源码

def build_targets(p, targets, model):
    # 这里输入的p包含了三个YoloLayer的输出,相比model里YoloLayer的输出多了一个维度
    # YoloLayer的p的shape为:(batch_size,anchor_num,grid_cell,grid_cell,xywh+obj_confidence+classes_num)
    # targets: [num_obj, 6] , that number 6 means -> (img_index, obj_index, x, y, w, h)

    # Build targets for compute_loss(), input targets(image,class,x,y,w,h)
    # shape[0]返回第一维度的个数,即image的个数,num_targets

    # 获取当前批次的target数
    nt = targets.shape[0]
    tcls, tbox, indices, anch = [], [], [], []
    # gain是一个6维的tensor
    gain = torch.ones(6, device=targets.device)  # normalized to gridspace gain

    multi_gpu = type(model) in (nn.parallel.DataParallel, nn.parallel.DistributedDataParallel)
    for i, j in enumerate(model.yolo_layers):  # model定义的yolo层索引list=[89, 101, 113]
        # 获取该第i个yolo predictor对应的anchors缩放后的尺度anchor_vec
        # anchor是shape为(3,2)的tensor,包含三组anchor尺度,分配给当前预测器
        anchors = model.module.module_list[j].anchor_vec if multi_gpu else model.module_list[j].anchor_vec
        # 2索引开始,即第三个元素开始到最后一个元素,{_,_,[],[],[],[]}其中[]为填充到gain的值
        gain[2:] = torch.tensor(p[i].shape)[[3, 2, 3, 2]]  # xyxy gain
        na = anchors.shape[0]  # number of anchors
        # [3] -> [3, 1] -> [3, nt]
        at = torch.arange(na).view(na, 1).repeat(1, nt)  # anchor tensor, same as .repeat_interleave(nt)

        # Match targets to anchors
        # gain的tensor状态为(1.,1.,grid_x,grid_y,grid_x,grid_y)其中在训练模式中,由于采用了多尺度训练,
        # gain中的grid_x和grid_y不一定相等,根据输入图像的size有关
        # targets的状态(img_index, obj_index, x, y, w, h)
        a, t, offsets = [], targets * gain, 0
        # t在这里t=targets*gain,解释:targets是图片的target归一化后的尺度,现在图片的feature map输出为(grid_x,grid_y)的尺度,
        # t则是将归一化的target映射到预测器的feature map上的尺度
        if nt:  # 如果存在target的话
            # iou_t = 0.20
            # j: [3, nt]
            # 传入anchors尺度,t为shape为(228,6)的tensor,取(228,(4,5))这个tensor传入,即w,h尺度
            # j是布尔值,大于0.2返回true,否则返回false,表示每组anchor和target的wh尺度,wh_iou表示宽高iou
            j = wh_iou(anchors, t[:, 4:6]) > model.hyp['iou_t']
            # iou(3,n) = wh_iou(anchors(3,2), gwh(n,2))
            # t.repeat(na, 1, 1): [nt, 6] -> [3, nt, 6]
            # 获取iou大于阈值的anchor与target对应信息
            # 这里非常重点,当前groundtruth的box信息和该预测器的其中一个anchor尺度的wh_iou>iou_t才能被筛选上
            a, t = at[j], t.repeat(na, 1, 1)[j]  # filter

        # Define
        # long等于to(torch.int64), 数值向下取整
        b, c = t[:, :2].long().T  # image_index, class_index
        gxy = t[:, 2:4]  # grid xy
        gwh = t[:, 4:6]  # grid wh
        gij = (gxy - offsets).long()  # 匹配targets所在的grid cell左上角坐标
        gi, gj = gij.T  # grid xy indices

        # Append
        # a为target使用的anchor索引
        indices.append((b, a, gj, gi))  # image, anchor, grid indices(x, y)
        tbox.append(torch.cat((gxy - gij, gwh), 1))  # gt box相对anchor的x,y偏移量以及w,h
        # anch为每个target使用的anchor尺度
        anch.append(anchors[a])  # anchors
        tcls.append(c)  # class
        if c.shape[0]:  # if any targets
            # 目标的标签数值不能大于给定的目标类别数
            assert c.max() < model.nc, 'Model accepts %g classes labeled from 0-%g, however you labelled a class %g. ' \
                                       'See https://github.com/ultralytics/yolov3/wiki/Train-Custom-Data' % (
                                           model.nc, model.nc - 1, c.max())
    # 返回值参数
    # 返回当前批次所有groundtruth中和anchor的wh_iou>iou_t这个超参的targets信息
    # tcls:筛选出来的gt的类索引
    # tbox:筛选出来的gt的box信息,tx,ty,w,h。其中tx,ty是偏移量;w,h是宽高
    # indics:(image_index,anchor_index,grid_y,grid_x)
    # anch:每个target对应使用的anchor尺度
    return tcls, tbox, indices, anch

build_targets源码详解

建议将上述源码复制到编译器后,边看代码边看解析

基础的参数解析

def build_targets(p, targets, model):

函数传入三个参数

  • p为model的输出,在build_target中只有一个作用,获取p的shape,然后将targets映射的p的shape尺度
  • targets为dataloader迭代器生成的一个batch的所有ground truth
  • model为整个yolo的model,以获取当前model对应YoloLayer的信息和YoloLayer对应的anchor尺度

p的shape为 ( Y o l o L a y e r _ n u m , b a t c h _ s i z e , a n c h o r _ n u m , g r i d _ y , g r i d _ x , x y w h + o b j _ c o n f i d e n c e + c l s _ n u m ) (YoloLayer\_num,batch\_size,anchor\_num,grid\_y,grid\_x,xywh+obj\_confidence+cls\_num) (YoloLayer_num,batch_size,anchor_num,grid_y,grid_x,xywh+obj_confidence+cls_num)

targets的shape为 ( n u m _ g r o u n d t r u t h , 6 ) (num\_groundtruth,6) (num_groundtruth,6),其中数字6代表 i m g _ i n d e x + c l s _ i n d e x + x y w h img\_index+cls\_index+xywh img_index+cls_index+xywh

    nt = targets.shape[0]
    tcls, tbox, indices, anch = [], [], [], []

nt获取targets第一个维度 n u m _ g r o u n d t r u t h num\_groundtruth num_groundtruth的数值
(groundtruth简写为gt)

  • tcls为筛选后gt的类索引
  • tbox为筛选后的gt的box信息,包含了 t x , t y , w , h t_x,t_y,w,h tx?,ty?,w,h信息,其中 t x t_x tx? t y t_y ty?为gt的中心坐标 x c e n t e r , y c e n t e r x_{center},y_{center} xcenter?,ycenter?离gt所在的 g r i d _ c e l l grid\_cell grid_cell的左上角坐标的偏移量, w , h w,h w,h表示gt的宽高信息
  • indices包含了tcls以及tbox信息的图像索引、所用的anchor索引、以及gt所在的 g r i d _ c e l l grid\_cell grid_cell信息,shape为 ( i m a g e _ i n d e x , a n c h o r _ i n d e x , g r i d _ y , g r i d _ x ) (image\_index,anchor\_index,grid\_y,grid\_x) (image_index,anchor_index,grid_y,grid_x)
  • anch为每个gt对应使用的anchor尺度

以上4组参数也是该build_targets所返回的参数

    gain = torch.ones(6, device=targets.device)  # normalized to gridspace gain
    multi_gpu = type(model) in (nn.parallel.DataParallel, nn.parallel.DistributedDataParallel)

前面提到buildd_targets的输入参数p的作用,这里gain的作用就是将输入参数p的shape转化为tensor,后面会提到gain的操作,这里只是对gain进行初始化,初始化为一个6维都为数值1的tensor

multi_gpu这里的处理代码我并没有了解,本人由于只使用了一个gpu,就没有去了解这相关的代码,感兴趣的请自行查阅相关库函数及处理

    for i, j in enumerate(model.yolo_layers):
        anchors = model.module.module_list[j].anchor_vec if multi_gpu else model.module_list[j].anchor_vec
        gain[2:] = torch.tensor(p[i].shape)[[3, 2, 3, 2]]
        na = anchors.shape[0]
        at = torch.arange(na).view(na, 1).repeat(1, nt) 
        a, t, offsets = [], targets * gain, 0

model.yolo_layers参数为model的成员变量,定义为yolo层索引list=[89, 101, 113]
anchor_vec为当前yolo_layer分配到的三组anchor缩放后的尺度,三个yolo_layer的缩放倍数分别为[32,16,8],这里的anchors表示当前yolo_layer的三组缩放后的anchor尺度。

上面提到的gain操作就是

gain[2:] = torch.tensor(p[i].shape)[[3, 2, 3, 2]]

p[i]表示第i个yolo_layer的输出,shape为 ( b s , a n c h o r , g r i d _ x , g r i d _ y , x y w h + o b j _ c o n f i d e n c e + c l s _ n u m ) (bs,anchor,grid\_x,grid\_y,xywh+obj\_confidence+cls\_num) (bs,anchor,grid_x,grid_y,xywh+obj_confidence+cls_num)
执行完上述代码之后,gain的值为 [ 1. , 1. , g r i d _ y , g r i d _ x , g r i d _ y , g r i d _ x ] [1.,1.,grid\_y,grid\_x,grid\_y,grid\_x] [1.,1.,grid_y,grid_x,grid_y,grid_x],待会以这个gain去让targets从一个归一化的值恢复到yolo_layer的feature map尺度上,这个feature map的尺度就是p的 g r i d _ c e l l grid\_cell grid_cell的尺度

na获取anchors第一个维度的值,即anchors的数量3
at生成一个shape为 ( 3 , n t ) (3,nt) (3,nt),其中 ( 1 , n t ) (1,nt) (1,nt)的值都为0, ( 2 , n t ) (2,nt) (2,nt)的值都为1, ( 3 , n t ) (3,nt) (3,nt)的值都为2.
更详细来看,我debug出来的结果如下,此时我当前batch的gt的数量,即nt为228,那么at的shape为 ( 3 , 228 ) (3,228) (3,228)
在这里插入图片描述
请添加图片描述

gt恢复到feature map尺度

a, t, offsets = [], targets * gain, 0

a作target使用的anchor索引用
gain的状态为: [ 1. , 1. , g r i d _ y , g r i d _ x , g r i d _ y , g r i d _ x ] [1.,1.,grid\_y,grid\_x,grid\_y,grid\_x] [1.,1.,grid_y,grid_x,grid_y,grid_x]
targets的shape为: ( n u m _ g r o u n d t r u t h , i m g _ i n d e x + c l s _ i n d e x + x + y + w + h ) (num\_groundtruth,img\_index+cls\_index+x+y+w+h) (num_groundtruth,img_index+cls_index+x+y+w+h)
gain与targets的 n u m _ g r o u n d t r u t h num\_groundtruth num_groundtruth个维度的 [ i m g _ i n d e x , c l s _ i n d e x , x , y , w , h ] [img\_index,cls\_index,x,y,w,h] [img_index,cls_index,x,y,w,h]tensor进行逐元素相乘,可将targets中所有gt的 x , y , w , h x,y,w,h x,y,w,h恢复到当前yolo_layer的feature map尺度上
注:后面提到的nt均表示 n u m _ g r o u n d t r u t h num\_groundtruth num_groundtruth

offsets默认为0,在获取当前gt所在的 g r i d _ c e l l grid\_cell grid_cell左上角坐标时会用到,但该函数offsets的设置一直为0,并没有什么作用

wh_IOU(宽高IOU筛选gt)

思路:gt与当前yolo_layer分配到的三组anchor进行宽高IOU筛选

        if nt:  # 如果存在target的话
            # iou_t = 0.20
            # j: [3, nt]
            # 传入anchors尺度,t为shape为(228,6)的tensor,取(228,(4,5))这个tensor传入,即w,h尺度
            # j是布尔值,大于0.2返回true,否则返回false,表示每组anchor和target的wh尺度,wh_iou表示宽高iou
            j = wh_iou(anchors, t[:, 4:6]) > model.hyp['iou_t']
            # iou(3,n) = wh_iou(anchors(3,2), gwh(n,2))
            # t.repeat(na, 1, 1): [nt, 6] -> [3, nt, 6]
            # 获取iou大于阈值的anchor与target对应信息
            # 这里非常重点,当前groundtruth的box信息和该预测器的其中一个anchor尺度的wh_iou>iou_t才能被筛选上
            a, t = at[j], t.repeat(na, 1, 1)[j]  # filter

这里判断是否有gt,有的话对gt进行筛选,筛选方式是宽高IOU,宽高IOU的源码如下:

def wh_iou(wh1, wh2):
    # Returns the nxm IoU matrix. wh1 is nx2, wh2 is mx2
    wh1 = wh1[:, None]  # [N,1,2]
    wh2 = wh2[None]  # [1,M,2]
    inter = torch.min(wh1, wh2).prod(2)  # [N,M]
    return inter / (wh1.prod(2) + wh2.prod(2) - inter)  # iou = inter / (area1 + area2 - inter)

None的作用是增加一个维度,具体请见注释
这里的宽高IOU与坐标形式的IOU计算有很大不同,这里的IOU对比的是尺度间的IOU关系。
请添加图片描述
如上图所示,宽高IOU的公式可以描述为:
w h _ I O U ( w h 1 , w h 2 ) = m i n { w 1 , w 2 } × m i n { h 1 , h 2 } w 1 h 1 + w 2 h 2 ? m i n { w 1 , w 2 } × m i n { h 1 , h 2 } wh\_IOU(wh_1,wh_2)=\frac{min\{ w_1,w_2\}\times min\{h_1,h_2\}}{w_1h_1+w_2h_2-min\{ w_1,w_2\}\times min\{h_1,h_2\}} wh_IOU(wh1?,wh2?)=w1?h1?+w2?h2??min{w1?,w2?}×min{h1?,h2?}min{w1?,w2?}×min{h1?,h2?}?
以此来描述两个box的尺度联系

j = wh_iou(anchors, t[:, 4:6]) > model.hyp['iou_t']

j表示对所有gt的box与三组anchor的wh_iou是否满足超参数iou_t=0.2的布尔关系
j是tensor为(3,nt)的参数,debug出来如下:
请添加图片描述

a, t = at[j], t.repeat(na, 1, 1)[j]  # filter

a前面已经说过,表示筛选后的gt的anchor索引
at前面已经说明了,at的shape为 ( 3 , n t ) (3,nt) (3,nt),其中 ( 1 , n t ) (1,nt) (1,nt)的值都为0, ( 2 , n t ) (2,nt) (2,nt)的值都为1, ( 3 , n t ) (3,nt) (3,nt)的值都为2.
注意:这里的筛选规则是,gt的box和其中一个anchor的尺度满足大于iou_t就被筛选上。
at[j]的作用:将j的三个维度中满足True的anchor索引筛选出来,对应的a的索引表示gt的索引。

t = t.repeat(na, 1, 1)[j]作用:前面提过t的shape为 ( n t , 6 ) (nt,6) (nt,6),即 ( n t , i m g _ i n d e x + c l s _ i n d e x + x + y + w + h ) (nt,img\_index+cls\_index+x+y+w+h) (nt,img_index+cls_index+x+y+w+h)
t.repeat(na,1,1)之后的shape为 ( 3 , n t , 6 ) (3,nt,6) (3,nt,6)第1、2和3维度的值均相同
经过t.repeat(na, 1, 1)[j]之后,筛选得到gt与a的索引对应
最终t的shape为 ( f i n a l _ g t _ n u m , 6 ) (final\_gt\_num,6) (final_gt_num,6),即 ( f i n a l _ g t _ n u m , i m g _ i n d e x + c l s _ i n d e x + x + y + w + h ) (final\_gt\_num,img\_index+cls\_index+x+y+w+h) (final_gt_num,img_index+cls_index+x+y+w+h)
注意:这里的 f i n a l _ g t _ n u m final\_gt\_num final_gt_num的数值大小范围为: 0 ≤ f i n a l _ g t _ n u m ≤ 3 × n t 0\leq final\_gt\_num\leq 3\times nt 0final_gt_num3×nt
这里筛选出来的gt对应的anchor并不是唯一的,一个gt至多对应三个anchor,这里只是初步经过wh_IOU筛选,传入compute_loss之后还会经过nms再次筛选

题外话:
这里在debug的时候,由于我的模型采用了多尺度训练,每次输出到yololayer的grid_cell都会不太一样,但调用的都是同一个batch的图片,当前batch的gt的数量是228个,对不同的grid_cell,筛选出来的gt也不同,从这里看出,grid_cell的大小也会影响筛选的gt数量,这可能是由于多尺度训练过程中对图片resize之后导致gt的实际信息和对应的anchor的wh_iou会发生变化,当然这只是我的猜测,留个坑,有兴趣的朋友可以研究下。

对筛选后的gt进行数值调整

        # Define
        # long等于to(torch.int64), 数值向下取整
        b, c = t[:, :2].long().T  # image_index, class_index
        gxy = t[:, 2:4]  # grid xy
        gwh = t[:, 4:6]  # grid wh
        gij = (gxy - offsets).long()  # 匹配targets所在的grid cell左上角坐标
        gi, gj = gij.T  # grid xy indices

t[:, :2].long().T对 t = ( n t , i m g _ i n d e x + c l s _ i n d e x + x + y + w + h ) t=(nt,img\_index+cls\_index+x+y+w+h) t=(nt,img_index+cls_index+x+y+w+h)的第二个维度开始筛选前两个值,即img_index和cls_index
b和c均为Tensor ( f i n a l _ g t _ n u m , ) (final\_gt\_num,) (final_gt_num,)均包含了 f i n a l _ g t _ n u m final\_gt\_num final_gt_num个值, 0 ≤ f i n a l _ g t _ n u m ≤ 3 × n t 0\leq final\_gt\_num\leq 3\times nt 0final_gt_num3×nt

记住,这里的t是已经恢复到yolo_layer的feature map尺度的tensor了
gxy = t[:, 2:4]:gxy获取x和y坐标,gwh同理

gij = (gxy - offsets).long() 这里offsets为0,等于没有使用到,这里long()函数将gxy向下取整,刚好就能得到当前gt的所在的grid的左上角坐标,具体解释如下图:
请添加图片描述
由上图可知, t x ∈ [ g i , g i + 1 ] , t y ∈ [ g j , g j + 1 ] t_x\in[g_i,g_i+1],t_y\in [g_j,g_j+1] tx?[gi?,gi?+1],ty?[gj?,gj?+1]
g x = g i + t x g_x=g_i+t_x gx?=gi?+tx?经过long()函数之后, t x t_x tx?被消除,剩下的 g i g_i gi?即当前gt的grid坐标

返回参数

        # Append
        # a为target使用的anchor索引
        indices.append((b, a, gj, gi))  # image, anchor, grid indices(x, y)
        tbox.append(torch.cat((gxy - gij, gwh), 1)) 
        # gt box相对anchor的x,y偏移量以及w,h
        # anch为每个target使用的anchor尺度
        anch.append(anchors[a])  # anchors
        tcls.append(c)  # class

tcls:筛选出来的gt的类索引
tbox:筛选出来的gt的box信息,tx,ty,w,h。其中tx,ty是偏移量;w,h是宽高
indics:(image_index,anchor_index,grid_y,grid_x)
anch:每个target对应使用的anchor尺度

总结

build_targets的代码比较繁琐,基本都是tensor的操作比较多,读起来不太容易,但搞清楚了这个函数,你就能知道获取的gt的状态,从而了解到loss是如何计算的,包括loss的正负样本的判别。

  人工智能 最新文章
2022吴恩达机器学习课程——第二课(神经网
第十五章 规则学习
FixMatch: Simplifying Semi-Supervised Le
数据挖掘Java——Kmeans算法的实现
大脑皮层的分割方法
【翻译】GPT-3是如何工作的
论文笔记:TEACHTEXT: CrossModal Generaliz
python从零学(六)
详解Python 3.x 导入(import)
【答读者问27】backtrader不支持最新版本的
上一篇文章      下一篇文章      查看所有文章
加:2021-08-06 09:37:50  更:2021-08-06 09:51:43 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/17 22:36:25-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码