人脸项目实践

1 前言

在实现一个优秀人脸识别系统的过程中,应该有以下重要环节。
1、干净而且大量的数据。
2、优秀的网络结构。
3、优秀的网络指挥棒(损失函数)。
4、由以上1,2,3决定一个优秀的模型,作为一个大型的分类过程,优秀的模型能够提取到人脸更加独有的特征,具有更好的“辨别特性”。
5、部署途径
(1)提取底图特征并保存数据库,就是要有每一个人的标准人脸图像,经过模型进行一遍提取之后的特征保存到数据库当中。
(2)索引匹配的人脸特征,对检测到的人进行检测、对齐矫正并经过模型提取到“独有”的特征,然后利用该特征与数据库中的人脸特征进行匹配,输出匹配程度最高的前几个人。

2 数据清洗

先用第三方的人脸识别对我的数据集进行一次特征提取和匹配,对于数据集中的每一个人我有一张标准的底图,提取了底图的特征并对每个人的所有图片进行匹配,将不匹配的图片剪切出来进行人工筛选,人多的话并合理用一些工具软件,用上几天就能将人脸数据集清洗得很好了,但不能随意删除图片,确认是不同的人或图片质量十分差才能清掉。

3 数据检测对齐矫正

数据检测对齐矫正,可以用ssh和dlib,也可以直接用作者提供的face_align.py,ssh表现效果确实比mtcnn这个好,但是相对麻烦了一点,所以我这里直接用了作者的align_dataset_mtcnn.py。

python face_align.py --source_root  E:/6T/data/test  --dest_root  E:/6T/data/test_Aligned

4 制作rec和idx的数据集

训练集需要.rec,.idx,*.lst,property,其中rec是已经对齐完成的图像数据,idx是索引,需要用到lst来生成rec和idx。
lst中包含的内容如下:1 path/Adam_Brody/Adam_Brody_277.png 25
中间用\t(TAB)隔开,用空格会报错,第一个参数代表是否对齐,第二个参数代表图片的路径,第三个参数代表图片的标签,整个lst文件要求标签必须从0并从小到大排列,不然生成的rec和idx会出错。
property是属性文件,里面内容是类别数和图像大小,例如 1000,112,112
其中1000代表人脸的类别数目,图片格式为112x112

4.1 生成.lst

import os
from easydict import EasyDict as edict

input_dir = 'E:\data\image'
ret = []
label = 0
person_names = []

for person_name in os.listdir(input_dir):
    person_names.append(person_name)

person_names = sorted(person_names)

for person_name in person_names:
    _subdir = os.path.join(input_dir, person_name)
    if not os.path.isdir(_subdir):
        continue

    _ret = []
    for img in os.listdir(_subdir):
        fimage = edict()
        fimage.id = os.path.join(_subdir, img)
        fimage.classname = str(label)
        fimage.image_path = os.path.join(_subdir, img)
        fimage.bbox = None
        fimage.landmark = None
        _ret.append(fimage)
    ret += _ret

    label += 1

for item in ret:
    print("%d\t%s\t%d" % (1, item.image_path, int(item.classname)))

手动生成:python dir2lst.py > E:\data\image.lst

4.2 由.lst 和 property 来生成 .rec .idx

手动编辑:property中就定义数据集的格式,整体内容如下:1000,112,112 ,其中1000代表人脸的类别数目,图片格式为112x112。

python face2rec2.py E:\data\

5 模型训练

训练的话最好写成一个脚本,直接运行脚本比较好修改参数。

configurations = {
    1: dict(
        SEED=1337,  # random seed for reproduce results

        DATA_ROOT='E:/6T/data/faces_glintasia',
        # the parent root where your train/val/test data are stored
        MODEL_ROOT='E:/6T/buffer/model',  # the root to buffer your checkpoints
        LOG_ROOT='E:/6T/buffer/log',  # the root to log your train/val status

        BACKBONE_RESUME_ROOT='./',  # the root to resume training from a saved checkpoint
        HEAD_RESUME_ROOT='./',  # the root to resume training from a saved checkpoint

        BACKBONE_NAME='IR_50',
        # support: ['ResNet_50', 'ResNet_101', 'ResNet_152', 'IR_50', 'IR_101', 'IR_152', 'IR_SE_50', 'IR_SE_101', 'IR_SE_152']
        HEAD_NAME='ArcFace',  # support:  ['Softmax', 'ArcFace', 'CosFace', 'SphereFace', 'Am_softmax']
        LOSS_NAME='Focal',  # support: ['Focal', 'Softmax']

        INPUT_SIZE=[112, 112],  # support: [112, 112] and [224, 224]
        RGB_MEAN=[0.5, 0.5, 0.5],  # for normalize inputs to [-1, 1]
        RGB_STD=[0.5, 0.5, 0.5],
        EMBEDDING_SIZE=512,  # feature dimension
        BATCH_SIZE=16,
        DROP_LAST=True,  # whether drop the last batch to ensure consistent batch_norm statistics
        LR=0.01,  # initial LR
        NUM_EPOCH=125,  # total epoch number (use the firt 1/25 epochs to warm up)
        WEIGHT_DECAY=5e-4,  # do not apply to batch_norm parameters
        MOMENTUM=0.9,
        STAGES=[20, 40, 60],  # epoch stages to decay learning rate

        DEVICE=torch.device("cuda:0" if torch.cuda.is_available() else "cpu"),
        MULTI_GPU=True,
        # flag to use multiple GPUs; if you choose to train with single GPU, you should first run "export CUDA_VISILE_DEVICES=device_id" to specify the GPU card you want to use
        GPU_ID=[0],  # specify your GPU ids
        PIN_MEMORY=True,
        NUM_WORKERS=0,
    ),
}

5 参考

https://github.com/ZhaoJ9014/face.evoLVe.PyTorch封装的网络框架
Insightface制作rec和idx的训练集
Insightface/Arcface项目实践流程

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352

推荐阅读更多精彩内容

  • 可能人类天生的利己主义思想在作怪,所以总是用双重标准对待别人。比如说,妈妈在打麻将的同时要求自己家的孩子读书,当孩...
    初入职场学语文阅读 456评论 4 3
  • 古人就是神奇,给我们留下了诸多的惊叹。这种惊叹,有明见的,也有不经挑明难以领略的,《易经》所云百姓曰用而不...
    木火通明文阅读 5,149评论 0 1
  • 日前,杭大中文系室友,浙江省永康籍知名诗人章锦水利用在兰溪参加金华市县(市、区)人大代表工委例会之机,参观方增先艺...
    陈水河阅读 801评论 3 15