R-CNN系列其五:R-FCN

作者: manofmountain | 来源:发表于2018-07-07 12:27 被阅读0次

    介绍

    R-CNN系列目标检测模型当中,发展到Faster-RCNN,它的模型检测准确率(及SOTA accuracy)已经达到了顶锋。直到目前为止尚没有一个目标检测模型在检测mAP大小上可以宣称能够在任一数据集上完胜于它。大家开始从其它方面来对它进一步提高,比如在检测时间上面。

    R-FCN模型同样提出于微软研究院的Sun,Jian团队(真是AI时代的弄潮儿啊)。它本质上任属于R-CNN系列的检测模型之一,因为它也是两阶段的检测网络,也需要先第一阶段训练得到RPN网络来提供下一阶段的ROIs,然后再进一步由ROIs与图片数据为输入,进一步地去训练检测阶段的网络。

    之前Fast-RCNN/Faster-RCNN的模型会将ROI pool层加在网络的中间,使得在ROI pool之后的每个层(其中包含有卷积层)都只在ROI范畴上进行处理,而非像ROI pool之间的层那样在整个图片范畴上进行特征处理。R-FCN与此不同,它的整个网络都是在对整个图片进行处理,只是到了模型的最后,在生成的feature maps之上加上ROI pooling层,然后再分别作分类与定位的回归处理。如此以来,因为它不需要存在某些层只针对ROI窗口处理,所以整体上它的检测速率更快,论文中作者实验证明在RPN平均生成2000个ROIs窗口时,R-FCN可比Faster-RCNN快20倍左右,同时能达到近似的检测mAP值。

    R-FCN的基本结构

    R-FCN模型同之前用于Semantic segmentation的FCN模型一样,整个模型当中并没有用到FC层,而是全部由convolution层构成。在模型的最后它生成出了一个特定的包含位置信息的score maps(这是本文最大的创新)。这个最后的scores maps共有k^2(C+1)个输出channels。其中C表示检测目标类别数目,k则表示最终ROI pool时所划分的空间节点数。这样最终的scores maps中会含有特定的第个类别的位置score值即每个具体的channel分别表示每个类所特定网格区域的值大小。

    它的基本概念可于下图得见。

    R-FCN用于目标检测的基本思想.JPG

    从图上可以看出,整个R-FCN模型只有到了最后才加入了ROI pooling层,而此ROI pooling在对scores maps做pooling时是对每个类之上的k^2个score maps分别按照其区域进行pooling操作即每次每个区域位置只在对应的一个score map上进行pooling处理(这是它模型最难理解的地方,当然也是它的最大创新即score maps上具有区域位置信息之所以能够有效的原因)。

    在下述表格中,作者对Faster-RCNN/R-CNN及R-FCN模型上ROI-wise处理的层数目做了比较,从理论上表明了R-FCN在做图片处理并最终进行目标检测时所可能具有的速率优势。

    R-FCN与其它R-CNN模型在方法论上的比较.JPG

    R-FCN详解

    上面章节我们讲述了R-FCN模型所包含的最重大的思想。下面我们则分别讲下R-FCN的基本网络结构组成及其中包含计算的特点。首先它的基本架构可见于下图。

    R-FCN模型的基本框架.JPG

    可以看出它本质上同其它R-CNN模型一样也是个需要训练出RPN网络的两阶段目标检测模型。另外在模型的最后它输出了一个包含k^2(C+1)个channels的score maps。在此最终的特征score maps上,再进行特殊的ROI pool处理。它的特殊性在于,一个ROI pool针对每个C之上的k^2个channels一起处理,最终生成出一个C channel出来(具有着固定的空间大小即k2)。而在此pooling处理当中,每个输出grid(i,j)的值都是在此k2个channel中的某一个即第(i * j)个channel上来完成的,有些绕。。真是需要细加领悟,才能明白(想想,这个Pool是在k^2 channels wise上来进行的,而每个output(i,j) = ROI_pool(channel(i*j)[i,j]))。

    然后R-FCN再对此ROI pool层输出的C+1个channels的score maps上进行一个普通的channel-wise的Pool处理(一般为average pool),最终得到一个包含C+1个值的特征相量。对此特征相量,我们可通过与lable值相对比,得出其softmax classification loss来。

    以上是类别检测的实现。对位置的检测是同样的。即在最终吐出的feature maps之上,它像之前Fast-RCNN/Faster-RCNN那样有一个并行的分支,用来对ROI的位置信息进行回归检测。因此位置检测分支先生成4 * K^2个score maps出来,然后同以上classification分支一样,对其做一个特殊的ROI pooling处理,然后再使用普通的average pool得到包含4个值的相量,接下来通过与位置信息的ground truth对比,算出反映位置信息的L1 loss出来。

    虽然文章中还有些细节实现上的tricks,像减少前端特征提取网络的总步长。然后使用Resnet101 base network的Conv4层,而非是改了步长的Conv5层等,还有其在Conv5层的filter之上所用的hole算法处理。但这些都是末节。。以上所讲才是它的主要概念。下面我们将分析下它这些基本概念的代码实现。

    代码分析

    R-FCN采用了多种基本CNN网络来提取特征。其中Resnet-101是最终效果最好的。它的100个中间层显然没啥好说的,跟论文所说无关。我们只需要关注下它的输入数据特点以及最终在主干网络生成的feature maps之上R-FCN是如何生成score maps以及做ROI pooling等分别进行classification 与localization detection的就可。

    以下首先是它的模型输入。这里我们假定已经有了train好的RPN网络所产生的ROIs了,它已是我们模型输入的一部分。此处会发现bbox_targets即反映检测框位置信息的值是8维度的,跟我们平时所以为的x,y,w,h四个值不同。那是因为R-CNN系列模型一直生成两种x,y,w,h的回归值出来,一组为背景,一组则为实际目标。

    name: "ResNet-101"
    
    input: "data"
    input_dim: 1
    input_dim: 3
    input_dim: 224
    input_dim: 224
    
    input: "rois"
    input_dim: 1 # to be changed on-the-fly to num ROIs
    input_dim: 5 # [batch ind, x1, y1, x2, y2] zero-based indexing
    input_dim: 1
    input_dim: 1
    
    input: "labels"
    input_dim: 1 # to be changed on-the-fly to match num ROIs
    input_dim: 1
    input_dim: 1
    input_dim: 1
    
    input: "bbox_targets"
    input_dim: 1  # to be changed on-the-fly to match num ROIs
    input_dim: 8 # 4 * (K+1) (=2) classes
    input_dim: 1
    input_dim: 1
    
    input: "bbox_loss_weights"
    input_dim: 1  # to be changed on-the-fly to match num ROIs
    input_dim: 8 # 4 * (K+1) (=2) classes
    input_dim: 1
    input_dim: 1
    

    再向下,我们看下它的score maps生成。可以看出,我们对Resnet-101的输出先是加了新的一个Conv层来将feature maps的channels数由2048降到了1024。然后在其上分别对目标类别与位置生成了包含有k^2 * (C+1)及 k^2 * ( 4 * 2)个channels的score maps。在这里模型使用的是VOC数据集,因此C为21,k则为7(亦可为其它,论文中作者还使用了3)。

    #----------------------new conv layer------------------
    layer {
        bottom: "res5c"
        top: "conv_new_1"
        name: "conv_new_1"
        type: "Convolution"
        convolution_param {
            num_output: 1024
            kernel_size: 1
            pad: 0
            weight_filler {
                type: "gaussian"
                std: 0.01
            }
            bias_filler {
                type: "constant"
                value: 0
            }
        }
        param {
            lr_mult: 1.0
        }
        param {
            lr_mult: 2.0
        }
    }
    
    layer {
        bottom: "conv_new_1"
        top: "conv_new_1"
        name: "conv_new_1_relu"
        type: "ReLU"
    }
    
    layer {
        bottom: "conv_new_1"
        top: "rfcn_cls"
        name: "rfcn_cls"
        type: "Convolution"
        convolution_param {
            num_output: 1029 #21*(7^2) cls_num*(score_maps_size^2)
            kernel_size: 1
            pad: 0
            weight_filler {
                type: "gaussian"
                std: 0.01
            }
            bias_filler {
                type: "constant"
                value: 0
            }
        }
        param {
            lr_mult: 1.0
        }
        param {
            lr_mult: 2.0
        }
    }
    layer {
        bottom: "conv_new_1"
        top: "rfcn_bbox"
        name: "rfcn_bbox"
        type: "Convolution"
        convolution_param {
            num_output: 392 #8*(7^2) cls_num*(score_maps_size^2)
            kernel_size: 1
            pad: 0
            weight_filler {
                type: "gaussian"
                std: 0.01
            }
            bias_filler {
                type: "constant"
                value: 0
            }
        }
        param {
            lr_mult: 1.0
        }
        param {
            lr_mult: 2.0
        }
    }
    

    按下来则是它的特殊ROI pooling的实现。此ROI pooling成功在类别检测分支上将k^2 ( C+ 1)个包含特定位置信息的输入channels 聚合为C + 1个输出channel。同时亦在区域检测分支上将k^2 * 8个包含位置信息的input channels聚会为8个output channel。

    #--------------position sensitive RoI pooling--------------
    layer {
        bottom: "rfcn_cls"
        bottom: "rois"
        top: "psroipooled_cls_rois"
        name: "psroipooled_cls_rois"
        type: "PSROIPooling"
        psroi_pooling_param {
            spatial_scale: 0.0625
            output_dim: 21
            group_size: 7
        }
    }
    
    layer {
        bottom: "psroipooled_cls_rois"
        top: "cls_score"
        name: "ave_cls_score_rois"
        type: "Pooling"
        pooling_param {
            pool: AVE
            kernel_size: 7
            stride: 7
        }
    }
    
    
    layer {
        bottom: "rfcn_bbox"
        bottom: "rois"
        top: "psroipooled_loc_rois"
        name: "psroipooled_loc_rois"
        type: "PSROIPooling"
        psroi_pooling_param {
            spatial_scale: 0.0625
            output_dim: 8
            group_size: 7
        }
    }
    
    layer {
        bottom: "psroipooled_loc_rois"
        top: "bbox_pred"
        name: "ave_bbox_pred_rois"
        type: "Pooling"
        pooling_param {
            pool: AVE
            kernel_size: 7
            stride: 7
        }
    }
    

    最终我们就得到了模型的输出即它在训练时所采用的loss值及中间检测模型效果的mAP值。

    #-----------------------output------------------------
    layer {
       name: "loss"
       type: "SoftmaxWithLoss"
       bottom: "cls_score"
       bottom: "labels"
       top: "loss_cls"
       loss_weight: 1
       propagate_down: true
       propagate_down: false
    }
    
    layer {
       name: "accuarcy"
       type: "Accuracy"
       bottom: "cls_score"
       bottom: "labels"
       top: "accuarcy"
       #include: { phase: TEST }
       propagate_down: false
       propagate_down: false
    }
    
    layer {
       name: "loss_bbox"
       type: "SmoothL1Loss"
       bottom: "bbox_pred"
       bottom: "bbox_targets"
       bottom: "bbox_loss_weights"
       top: "loss_bbox"
       loss_weight: 1
       propagate_down: true
       propagate_down: false
       propagate_down: false
    }
    

    参考文献

    相关文章

      网友评论

        本文标题:R-CNN系列其五:R-FCN

        本文链接:https://www.haomeiwen.com/subject/jahwuftx.html