README.md 6.2 KB
Newer Older
B
Bai Yifan 已提交
1
>运行该示例前请安装PaddleSlim和Paddle1.6或更高版本
2 3 4 5 6

# 检测模型蒸馏示例

## 概述

B
Bai Yifan 已提交
7
该示例使用PaddleSlim提供的[蒸馏策略](https://paddlepaddle.github.io/PaddleSlim/algo/algo/#3)对检测库中的模型进行蒸馏训练。
8 9
在阅读该示例前,建议您先了解以下内容:

10
- [检测库的常规训练方法](https://github.com/PaddlePaddle/PaddleDetection)
B
Bai Yifan 已提交
11
- [PaddleSlim蒸馏API文档](https://paddlepaddle.github.io/PaddleSlim/api/single_distiller_api/)
12

B
Bai Yifan 已提交
13 14
## 安装PaddleSlim
可按照[PaddleSlim使用文档](https://paddlepaddle.github.io/PaddleSlim/)中的步骤安装PaddleSlim
15

B
Bai Yifan 已提交
16
## 蒸馏策略说明
17

B
Bai Yifan 已提交
18
关于蒸馏API如何使用您可以参考PaddleSlim蒸馏API文档
19

B
Bai Yifan 已提交
20
这里以ResNet34-YoloV3蒸馏训练MobileNetV1-YoloV3模型为例,首先,为了对`student model``teacher model`有个总体的认识,进一步确认蒸馏的对象,我们通过以下命令分别观察两个网络变量(Variables)的名称和形状:
21 22

```python
B
Bai Yifan 已提交
23 24
# 观察student model的Variables
student_vars = []
25
for v in fluid.default_main_program().list_vars():
B
Bai Yifan 已提交
26 27 28 29 30 31 32 33
    try:
        student_vars.append((v.name, v.shape))
    except:
        pass
print("="*50+"student_model_vars"+"="*50)
print(student_vars)
# 观察teacher model的Variables
teacher_vars = []
34
for v in teacher_program.list_vars():
B
Bai Yifan 已提交
35 36 37 38 39 40
    try:
        teacher_vars.append((v.name, v.shape))
    except:
        pass
print("="*50+"teacher_model_vars"+"="*50)
print(teacher_vars)
41 42
```

B
Bai Yifan 已提交
43
经过对比可以发现,`student model``teacher model`输入到3个`yolov3_loss`的特征图分别为:
44 45 46

```bash
# student model
B
Bai Yifan 已提交
47
conv2d_20.tmp_1, conv2d_28.tmp_1, conv2d_36.tmp_1
48
# teacher model
B
Bai Yifan 已提交
49
conv2d_6.tmp_1, conv2d_14.tmp_1, conv2d_22.tmp_1
50 51 52
```


B
Bai Yifan 已提交
53 54 55 56 57 58
它们形状两两相同,且分别处于两个网络的输出部分。所以,我们用`l2_loss`对这几个特征图两两对应添加蒸馏loss。需要注意的是,teacher的Variable在merge过程中被自动添加了一个`name_prefix`,所以这里也需要加上这个前缀`"teacher_"`,merge过程请参考[蒸馏API文档](https://paddlepaddle.github.io/PaddleSlim/api/single_distiller_api/#merge)

```python
dist_loss_1 = l2_loss('teacher_conv2d_6.tmp_1', 'conv2d_20.tmp_1')
dist_loss_2 = l2_loss('teacher_conv2d_14.tmp_1', 'conv2d_28.tmp_1')
dist_loss_3 = l2_loss('teacher_conv2d_22.tmp_1', 'conv2d_36.tmp_1')
59 60
```

B
Bai Yifan 已提交
61
我们也可以根据上述操作为蒸馏策略选择其他loss,PaddleSlim支持的有`FSP_loss`, `L2_loss`, `softmax_with_cross_entropy_loss` 以及自定义的任何loss。
62 63 64

## 训练

B
Bai Yifan 已提交
65 66
根据[PaddleDetection/tools/train.py](../../tools/train.py)编写压缩脚本`distill.py`
在该脚本中定义了teacher_model和student_model,用teacher_model的输出指导student_model的训练
67

B
Bai Yifan 已提交
68
### 执行示例
69

B
Bai Yifan 已提交
70
step1: 设置GPU卡
71

B
Bai Yifan 已提交
72 73 74
```shell
export CUDA_VISIBLE_DEVICES=0,1,2,3,4,5,6,7
```
75

B
Bai Yifan 已提交
76
step2: 开始训练
77

B
Bai Yifan 已提交
78 79 80 81 82 83
```bash
python slim/distillation/distill.py \
    -c configs/yolov3_mobilenet_v1_voc.yml \
    -t configs/yolov3_r34_voc.yml \
    --teacher_pretrained https://paddlemodels.bj.bcebos.com/object_detection/yolov3_r34_voc.tar
```
84

B
Bai Yifan 已提交
85
如果要调整训练卡数,需要调整配置文件`yolov3_mobilenet_v1_voc.yml`中的以下参数:
86

B
Bai Yifan 已提交
87 88 89 90 91
- **max_iters:** 训练过程迭代总步数。
- **YOLOv3Loss.batch_size:** 该参数表示单张GPU卡上的`batch_size`, 总`batch_size`是GPU卡数乘以这个值, `batch_size`的设定受限于显存大小。
- **LeaningRate.base_lr:** 根据多卡的总`batch_size`调整`base_lr`,两者大小正相关,可以简单的按比例进行调整。
- **LearningRate.schedulers.PiecewiseDecay.milestones:** 请根据batch size的变化对其调整。
- **LearningRate.schedulers.PiecewiseDecay.LinearWarmup.steps:** 请根据batch size的变化对其进行调整。
92

B
Bai Yifan 已提交
93
以下为4卡训练示例,通过命令行覆盖`yolov3_mobilenet_v1_voc.yml`中的参数:
94

B
Bai Yifan 已提交
95 96 97 98 99 100 101 102
```shell
CUDA_VISIBLE_DEVICES=0,1,2,3
python slim/distillation/distill.py \
    -c configs/yolov3_mobilenet_v1_voc.yml \
    -t configs/yolov3_r34_voc.yml \
    -o YoloTrainFeed.batch_size=16 \
    --teacher_pretrained https://paddlemodels.bj.bcebos.com/object_detection/yolov3_r34_voc.tar
```
103 104 105 106




B
Bai Yifan 已提交
107
### 保存断点(checkpoint)
108

B
Bai Yifan 已提交
109
蒸馏任务执行过程中会自动保存断点。如果需要从断点继续训练请用`-r`参数指定checkpoint路径,示例如下:
110

B
Bai Yifan 已提交
111 112 113 114 115 116 117
```bash
python -u slim/distillation/distill.py \
-c configs/yolov3_mobilenet_v1_voc.yml \
-t configs/yolov3_r34_voc.yml \
-r output/yolov3_mobilenet_v1_voc/10000 \
--teacher_pretrained https://paddlemodels.bj.bcebos.com/object_detection/yolov3_r34_voc.tar
```
118 119 120 121




B
Bai Yifan 已提交
122 123 124
## 评估

每隔`snap_shot_iter`步后会保存一个checkpoint模型可以用于评估,使用PaddleDetection目录下[tools/eval.py](../../tools/eval.py)评估脚本,并指定`weights`为训练得到的模型路径
125 126

运行命令为:
B
Bai Yifan 已提交
127 128 129 130
```bash
export CUDA_VISIBLE_DEVICES=0
python -u tools/eval.py -c configs/yolov3_mobilenet_v1_voc.yml \
       -o weights=output/yolov3_mobilenet_v1_voc/model_final \
131 132
```

B
Bai Yifan 已提交
133
## 预测
134

B
Bai Yifan 已提交
135
每隔`snap_shot_iter`步后保存的checkpoint模型也可以用于预测,使用PaddleDetection目录下[tools/infer.py](../../tools/infer.py)评估脚本,并指定`weights`为训练得到的模型路径
136

B
Bai Yifan 已提交
137
### Python预测
138

B
Bai Yifan 已提交
139 140 141 142 143 144 145 146 147
运行命令为:
```
export CUDA_VISIBLE_DEVICES=0
python -u tools/infer.py -c configs/yolov3_mobilenet_v1_voc.yml \
                    --infer_img=demo/000000570688.jpg \
                    --output_dir=infer_output/ \
                    --draw_threshold=0.5 \
                    -o weights=output/yolov3_mobilenet_v1_voc/model_final
```
148

B
Bai Yifan 已提交
149
## 示例结果
150

B
Bai Yifan 已提交
151
### MobileNetV1-YOLO-V3-VOC
152

B
Bai Yifan 已提交
153 154 155 156 157 158 159 160
| FLOPS |输入尺寸|每张GPU图片个数|推理时间(fps)|Box AP|下载|
|:-:|:-:|:-:|:-:|:-:|:-:|
|baseline|608     |16|104.291|76.2|[下载链接](https://paddlemodels.bj.bcebos.com/object_detection/yolov3_mobilenet_v1_voc.tar)|
|baseline|416 |16|-|76.7|[下载链接](https://paddlemodels.bj.bcebos.com/object_detection/yolov3_mobilenet_v1_voc.tar)|
|baseline|320 |16|-|75.3|[下载链接](https://paddlemodels.bj.bcebos.com/object_detection/yolov3_mobilenet_v1_voc.tar)|
|蒸馏后|608 |16|106.914|79.0||
|蒸馏后|416 |16|-|78.2||
|蒸馏后|320 |16|-|75.5||
161

B
Bai Yifan 已提交
162
> 蒸馏后的结果用ResNet34-YOLO-V3做teacher,4GPU总batch_size64训练90000 iter得到