• 【人脸识别】MVFace:一个优于CosFace和ArcFace的人脸识别损失


    论文题目:《Mis-classifified Vector Guided Softmax Loss for Face Recognition》
    论文地址:https://arxiv.org/pdf/1912.00833v1.pdf
    代码地址:http://www.cbsr.ia.ac.cn/users/xiaobowang/

    1.背景

           迄今为止,提出了几种基于margin的softmax损失函数(cosface、sphereface、arcface…)用来增加不同类特征的间隔。虽然它们都取得了重要的成就,但是也存在几个问题:
    1、没有考虑到特征挖掘的重要性,而且样本挖掘的时候怎么清晰定义难易样本;
    2、设置margin时只考虑从Ground Truth 类出发,未从其他类别考虑判别性;
    3、设置margin时不同的类都是固定的间隔m值,不适合很多真实场景的情况;
           MVface可以自适应地重点关注误分类的特征向量,以此指导判别性特征学习。这也是首次将特征margin的优点和特征挖掘的优点集中在一个统一的损失函数里面。

    2.相关工作

           为了解决问题1,曾经有人采用基于样本挖掘策略的softmax损失函数(HM-softmax),在mini-batch选取一定比例的高损失的样本来训练,但是这个困难样本的比例往往取决于经验,简单样本则完全丢弃。后来有人设计了一个相对’软’的策略,名为Focal loss(F-softmax),也就是简单样本不用丢弃,只是轻微关注,而重点关注困难样本。然而这些困难样本的定义是不清晰的,所以这两个策略难以提升性能。

    • softmax损失函数:
      在这里插入图片描述
      K是类别数,wk权重和特征x经过归一化后被参数s取代
    • Mining-based Softmax函数:
      在这里插入图片描述
      Py是预测的后验概率,g(Py)是一个指示函数
      如果是HM-Softmax, g(Py)=1 if sample is hard else 0
      如果是F-Softmax, g(Py)=(1-Py)𝛾
    • Margin-based Softmax损失函数:
      在这里插入图片描述
      f(m, 𝜃wy, x)是设计的Margin函数,有多种形式,比如A-Softmax,AM-softmax、Arc-Softmax都有不同的设计。甚至还可以将他们混合起来为 f(m,𝜃wy,x)=cos(m1, 𝜃wy, x+m3)-m2
    • Naive Mining-Margin Softmax Loss
      为了解决问题1,最直接简单的手段将Mining-based 和Margin-based 集中在一起
      在这里插入图片描述
      但是在实践中上述函数并没有很好地改进性能,原因可能是HM-Softmax舍弃了部分容易样本,而对于F-Softmax而言,它使用了所有样本,并根据经验通过调整因子对每个样本进行了加权,但是训练中的困难样本定义不清晰,没有直观的解释。

    3.MVFace

    直觉说,考虑分离良好的特征向量对学习问题影响很小。 这意味着错误分类的特征向量对于增强特征判别性更为关键。论文中定义一个指示器函数动态地指定一个样本是否为误分类,如下:

    在这里插入图片描述

    k不等于y,也就是Ik与除了GroundTruth的其他类有关。

    在这里插入图片描述

    比如看上图的左边部分,特征x2属于类1,但是被误分类了,即f(m, 𝜃w1,x2)-cos(𝜃w2,x2)<0,那么这些样本会暂时被重点对待,用这种方式困难样本就清晰地被指示了。主要是集中这些困难样本的训练,因此制定了MV-Softmax损失函数:

    在这里插入图片描述

    h(t,𝜃wy,x,Ik)是对误分类的样本加权的函数,有两种形式,一个是对所有误分的类固定权重:

    在这里插入图片描述

    另外一种是自适应动态加权:

    在这里插入图片描述

    t>=0是预设超参数,很明显,如果t=0就成为了Margin-based Softmax损失函数
    Empirically set t in [0.2,0.3]

    最终损失(自适应):

    在这里插入图片描述

    4. 对比

    4.1. Comparision to Mining-based Softmax Losses

    在这里插入图片描述
           假设两个样本x1和x2 都属于类别1,x1可以很好的分类但x2出现误分类情况。HM-Softmax丢弃了简单样本x1而用困难样本x2训练。F-Softmax并未明确指示困难样本,但会重新加权所有样本,从而使一个较困难的x2具有相对较大的损失值。这两种策略都是直接从损失的角度出发,困难样本的选择没有语义上的指导。本文提出的MV-softmax首先根据决策边界在语义上定义了困难样本,而且是从概率的角度来关注困难样本。由于-log§是单调递减函数,减少误分类向量x2的后验概率将会增加x2在训练中的重要性。总的来说可以看出本文提出的方法对于判别性特征学习更优于先前的方法。

    4.2. Comparision to Margin-based Softmax Losses

           依然假设样本x2来自于类别1,没有很好的分类。原始的softmax损失目标是使得w1T*x2 > w2Tx2,w1Tx2 > w3Tx3,特征和权重归一化后等价为:cos(𝜃1) > cos(𝜃2),cos(𝜃1) > cos(𝜃3)。为了使上述关系更加严格,基于margin的损失函数引进一个margin函数f(m,𝜃1)=cos(m1𝜃1+m3)-m2,这个margin函数是从GroundTruth类出发,只与𝜃1有关。

    在这里插入图片描述

           f(m,𝜃1)对于不同的类都是相同固定的,忽视了与其他类的判别性,为了解决这个问题,本文针对误分类的x2引入了与其他类有关的margin函数h*(t,𝜃2)

    在这里插入图片描述

           h*(t,𝜃2) = cos(𝜃2) + t (固定形式) or (t + 1)cos(𝜃2) + t (自适应),对于𝜃3而言,由于x2被w3很好的分类,所以不需要加上额外的条件。用MV-AM-Softmax举例 (f(m,𝜃y) = cos(𝜃y) - m),对于误分类的样本,margin为m + t or m + tcos(𝜃2) + t(这个margin是自适应的与cos(𝜃2)有关 )。通过以上的这些改进,MV-Softmax很好地解决了第二和第三个缺点。

    5. 算法流程

    在这里插入图片描述

    MVArcFace

    class MVArcFace(nn.Module):
        # Mis-classified Vector Guided Softmax Loss  CVPR 2019
        def __init__(self, in_features, out_features, device_id=None, s = 32.0, m = 0.35, t=0.2, easy_margin = False, fp16 = False):
            super(MVArcFace, self).__init__()
            self.in_features = in_features
            self.out_features = out_features
            self.device_id = device_id
    
            self.s = s
            self.m = m
            self.t = t
    
            self.weight = Parameter(torch.FloatTensor(out_features, in_features))
            nn.init.xavier_uniform_(self.weight)
    
            self.easy_margin = easy_margin
            self.cos_m = math.cos(m)
            self.sin_m = math.sin(m)
            self.th = math.cos(math.pi - m)
            self.mm = math.sin(math.pi - m) * m
            self.fp16 = fp16
    
        def forward(self, input, label):
            # --------------------------- cos(theta) & phi(theta) ---------------------------
            if self.device_id == None:
                cos_theta = F.linear(F.normalize(input), F.normalize(self.weight))
            else:
                x = input
                sub_weights = torch.chunk(self.weight, len(self.device_id), dim=0)
                temp_x = x.cuda(self.device_id[0])
                weight = sub_weights[0].cuda(self.device_id[0])
                cos_theta = F.linear(F.normalize(temp_x), F.normalize(weight))
                for i in range(1, len(self.device_id)):
                    temp_x = x.cuda(self.device_id[i])
                    weight = sub_weights[i].cuda(self.device_id[i])
                    cos_theta = torch.cat((cos_theta, F.linear(F.normalize(temp_x), F.normalize(weight)).cuda(self.device_id[0])), dim=1)
            #cosɵ:[-1,1] (320,181754)
            cos_theta = cos_theta.clamp(-1, 1)
            batch_size = label.size(0)
            #gt cosɵy (1,181754)
            gt = cos_theta[torch .arange(0, batch_size), label].view(-1, 1)
            #gt sinɵy (1,181754)
            sin_theta = torch.sqrt(1.0 - torch.pow(gt, 2))
            #gt cos(ɵy+m) (1,181754)
            cos_theta_m = gt * self.cos_m - sin_theta * self.sin_m
            
            mask = cos_theta > cos_theta_m  #cosɵj > gt cos(ɵ+m) (320,181754)
            hard_vector = cos_theta[mask]   
            cos_theta[mask] = (self.t + 1) * hard_vector + self.t #(t+1)cosɵj+t
    
            if self.fp16:
                cos_theta_m = cos_theta_m.half()
            if self.easy_margin:
                final_gt = torch.where(gt > 0.0, cos_theta_m, gt)
            else:
                final_gt = torch.where(gt > self.th, cos_theta_m, gt - self.mm) #角度判断能否+m
            if self.device_id != None:
                cos_theta = cos_theta.cuda(self.device_id[0])
            cos_theta.scatter_(1, label.data.view(-1, 1), final_gt)
            cos_theta *= self.s
    
            return cos_theta
    
    • 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
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62

    MVSoftmax

    class MV_Softmax(nn.Module):
        """Implementation for "Mis-classified Vector Guided Softmax Loss for Face Recognition"
        """
        def __init__(self, in_features, out_features, device_id=None, is_am=True, margin=0.35, mv_weight=1.2, scale=32, fp16 = False):
            super(MV_Softmax, self).__init__()
            self.device_id = device_id
            #self.weight = Parameter(torch.Tensor(in_features, out_features))
            #self.weight.data.uniform_(-1, 1).renorm_(2, 1, 1e-5).mul_(1e5)
            self.weight = Parameter(torch.FloatTensor(out_features, in_features))
            nn.init.xavier_uniform_(self.weight)
    
            self.margin = margin
            self.mv_weight = mv_weight
            self.scale = scale
            self.is_am = is_am
            self.cos_m = math.cos(margin)
            self.sin_m = math.sin(margin)
            self.threshold = math.cos(math.pi - margin)
            self.mm = self.sin_m * margin
    
        def forward(self, x, label):
            #kernel_norm = F.normalize(self.weight, dim=0)
            #x = F.normalize(x)
            #cos_theta = torch.mm(x, kernel_norm)
            sub_weights = torch.chunk(self.weight, len(self.device_id), dim=0)
            temp_x = x.cuda(self.device_id[0])
            weight = sub_weights[0].cuda(self.device_id[0])
            cos_theta = F.linear(F.normalize(temp_x), F.normalize(weight))
            for i in range(1, len(self.device_id)):
                temp_x = x.cuda(self.device_id[i])
                weight = sub_weights[i].cuda(self.device_id[i])
                cos_theta = torch.cat((cos_theta, F.linear(F.normalize(temp_x), F.normalize(weight)).cuda(self.device_id[0])), dim=1)
            batch_size = label.size(0)
            gt = cos_theta[torch.arange(0, batch_size), label].view(-1, 1) 
            if self.is_am:  # AM
                mask = cos_theta > gt - self.margin
                final_gt = torch.where(gt > self.margin, gt - self.margin, gt)
            else:  # arcface
                sin_theta = torch.sqrt(1.0 - torch.pow(gt, 2))
                cos_theta_m = gt * self.cos_m - sin_theta * self.sin_m 
                mask = cos_theta > cos_theta_m
                final_gt = torch.where(gt > 0.0, cos_theta_m, gt)
            # process hard example.
            hard_example = cos_theta[mask]
            cos_theta[mask] = self.mv_weight * hard_example + self.mv_weight - 1.0
            if self.device_id != None:
                cos_theta = cos_theta.cuda(self.device_id[0])
            cos_theta.scatter_(1, label.data.view(-1, 1), final_gt)
            cos_theta *= self.scale
            return cos_theta
    
    • 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
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
  • 相关阅读:
    【ArcGIS Pro二次开发】(71):添加图层名称和路径到字段
    视口 css
    WPF路由事件
    python二次开发CATIA:新建Part文档
    蓝桥杯算法竞赛第一周题型总结
    openwrt上/etc/localtime报错问题解决
    一口气说出4种主流数据库ID自增长,面试官懵了
    MAC和安卓手机文件传输OpenMTP
    springcloud:四、nacos介绍+启动+服务分级存储模型/集群+NacosRule负载均衡
    C++类型安全
  • 原文地址:https://blog.csdn.net/Roaddd/article/details/128179310