Shortcuts

教程 1: 了解配置文件

MMSelfSup 主要是在 python 文件中来设置各种各样的配置。我们配置文件系统的设计融合了模块化和可继承的设计理念,可以让用户轻松方便地完成各种实验配置。所有的配置文件全部位于 configs 目录下。如果您想查看配置文件的全貌,您可以使用以下命令 python tools/misc/print_config.py

配置文件命名规则

我们使用以下规则来命名我们的配置文件,社区贡献者建议遵循这个规则来贡献您的代码。简单来说,配置文件的名字主要划分为四个部分: algorithm info, module information, training informationdata information。不同部分通过下划线 _ 来进行相连,而属于同一个部分的内容,通过中横线 -来进行相连。

我们使用以下一个实例让大家有一个清晰的认识

{algorithm_info}_{module_info}_{training_info}_{data_info}.py
  • algorithm_info:与算法相关的一些信息,例如算法名;

  • module_info: 模块相关的一些信息,例如与loss, head相关的信息;

  • training_info:训练相关的信息, 例如 batch size, 学习率调整器和数据增强策略。

  • data_info:数据相关信息, 例如数据集名,输入图片的大小;

在下面几个章节,我们将对文件名中的各个部分进行详细的说明:

算法信息

{algorithm}-{misc}

algorithm 通常情况下是算法名字的缩写和版本号. 例如:

  • relative-loc : 算法名中不同的部分通过中横线 -相连

  • simclr

  • mocov2

misc 描述了算法的一些其他信息

  • npid-ensure-neg

  • deepcluster-sobel

模块信息

{backbone_setting}-{neck_setting}-{head_setting}-{loss_setting}

模块信息大部分情况下是有关 backbone 的一些信息. 例如:

  • resnet50

  • vit-base-p16

  • swin-base

有时候,有些特殊的配置需要在配置文件名中提及,例如:

  • resnet50-sobel: 在诸如线性评测之类的下游任务, 当我们使用的是 DeepCluster 的预训练模型,在经过 Sobel 层之后,模型只接受两层输入

neck_setting, head_settingloss_setting 这几个选项是可选的。

训练信息

训练相关的一些配置,包括 batch size, 学习率调整方案和数据增强等。

  • Batch size, 其格式为 {gpu x batch_per_gpu},如 8xb32;

  • 训练配置, 他们需要以下面这个格式来进行书写 {pipeline aug}-{train aug}-{scheduler}-{epochs}

如:

  • 8xb32-mcrop-2-6-coslr-200e : mcrop 是 SwAV 提出的 pipeline 中的名为 multi-crop 的一部分。2 和 6 表示 2 个 pipeline 分别输出 2 个和 6 个裁剪图,而且裁剪信息记录在数据信息中;

  • 8xb32-accum16-coslr-200e : accum16 表示权重会在梯度累积16个迭代之后更新。

  • 8xb512-amp-coslr-300e : amp 表示使用混合精度训练。

数据信息

数据信息包含数据集,输入大小等。例如:

  • in1k : ImageNet1k 数据集,默认使用的输入图像大小是 224x224

  • in1k-384px : 表示输入图像大小是384x384

  • cifar10

  • inat18 : iNaturalist2018 数据集,包含 8142 类

  • places205

配置文件命名示例

这一节,我们通过一个具体的例子来说明文件命名的规则:

swav_resnet50_8xb32-mcrop-2-6-coslr-200e_in1k-224-96.py
  • swav: 算法信息

  • resnet50: 模块信息

  • 8xb32-mcrop-2-6-coslr-200e: 训练信息

    • 8xb32: 共使用 8 张 GPU,每张 GPU 上的 batch size 是 32

    • mcrop-2-6: 使用 multi-crop 数据增强方法

    • coslr: 使用余弦学习率调度器

    • 200e: 训练模型200个周期

  • in1k-224-96: 数据信息,在 ImageNet1k 数据集上训练,输入大小是 224x224 和 96x96

配置文件结构

configs/_base_ 文件夹中, 有 4 种类型的基础组件文件,即:

  • models

  • datasets

  • schedules

  • runtime

所有的基础配置文件定义了训练所需的最基础的元素,例如 train/val/test 循环,优化器。你可以通过继承一些基础配置文件快捷地构建你自己的配置。由 _base_ 下的组件组成的配置被称为 原始配置(primitive)。为了易于理解,我们使用 MoCo v2 作为一个例子,并对它的每一行做出注释。若想了解更多细节,请参考 API 文档。

配置文件 configs/selfsup/mocov2/mocov2_resnet50_8xb32-coslr-200e_in1k.py 如下所述:

_base_ = [
    '../_base_/models/mocov2.py',                  # 模型
    '../_base_/datasets/imagenet_mocov2.py',       # 数据
    '../_base_/schedules/sgd_coslr-200e_in1k.py',  # 训练调度
    '../_base_/default_runtime.py',                # 运行时设置
]

# 我们继承了默认的运行时设置,同时修改了  ``CheckpointHook``.
# max_keep_ckpts 控制在 work_dirs 中最多保存多少个 checkpoint 文件
# 例如是 3, ``CheckpointHook`` 将会只保存最近的 3 个 checkpoint 文件
# 如果在 work_dirs 中超过了 3 个文件, 将会自动删掉时间最久远的那个 checkpoint
# , 从而保持 checkpoint 文件的数目始终为 3
default_hooks = dict(checkpoint=dict(max_keep_ckpts=3))

../_base_/models/mocov2.py 是 MoCo v2 的基础模型配置。

# type='MoCo' 指代我们使用 MoCo 这个算法。 我们将改算法分为四个部分:
# backbone, neck, head 和 loss。'queue_len', 'feat_dim' and 'momentum' 是另外
# 几个 MoCo 需要的参数。
model = dict(
    type='MoCo',
    queue_len=65536,
    feat_dim=128,
    momentum=0.999,
    data_preprocessor=dict(
        mean=(123.675, 116.28, 103.53),
        std=(58.395, 57.12, 57.375),
        bgr_to_rgb=True),
    backbone=dict(
        type='ResNet',
        depth=50,
        in_channels=3,
        out_indices=[4],  # 0: conv-1, x: stage-x
        norm_cfg=dict(type='BN')),
    neck=dict(
        type='MoCoV2Neck',
        in_channels=2048,
        hid_channels=2048,
        out_channels=128,
        with_avg_pool=True),
    head=dict(
        type='ContrastiveHead',
        loss=dict(type='mmcls.CrossEntropyLoss'),
        temperature=0.2))

../_base_/datasets/imagenet_mocov2.py 是 MoCo v2 的基础数据集配置。主要写出了 与 dataset 和 dataloader 相关的信息。

# dataset 配置
# 我们使用 MMClassification 中实现的 ``ImageNet`` dataset 数据集, 所以
# 这里有一个 ``mmcls`` 前缀.
dataset_type = 'mmcls.ImageNet'
data_root = 'data/imagenet/'

# mocov2 和 mocov1 的主要差异在于数据增强的不同
view_pipeline = [
    dict(
        type='RandomResizedCrop', size=224, scale=(0.2, 1.), backend='pillow'),
    dict(
        type='RandomApply',
        transforms=[
            dict(
                type='ColorJitter',
                brightness=0.4,
                contrast=0.4,
                saturation=0.4,
                hue=0.1)
        ],
        prob=0.8),
    dict(
        type='RandomGrayscale',
        prob=0.2,
        keep_channels=True,
        channel_weights=(0.114, 0.587, 0.2989)),
    dict(type='RandomGaussianBlur', sigma_min=0.1, sigma_max=2.0, prob=0.5),
    dict(type='RandomFlip', prob=0.5),
]

train_pipeline = [
    dict(type='LoadImageFromFile'),
    dict(type='MultiView', num_views=2, transforms=[view_pipeline]),
    dict(type='PackSelfSupInputs', meta_keys=['img_path'])
]

train_dataloader = dict(
    batch_size=32,
    num_workers=8,
    drop_last=True,
    persistent_workers=True,
    sampler=dict(type='DefaultSampler', shuffle=True),
    collate_fn=dict(type='default_collate'),
    dataset=dict(
        type=dataset_type,
        data_root=data_root,
        ann_file='meta/train.txt',
        data_prefix=dict(img_path='train/'),
        pipeline=train_pipeline))

../_base_/schedules/sgd_coslr-200e_in1k.py 是 MoCo v2 的基础调度配置。

# 优化器
optimizer = dict(type='SGD', lr=0.03, weight_decay=1e-4, momentum=0.9)
optim_wrapper = dict(type='OptimWrapper', optimizer=optimizer)

# 学习率调整策略
# 使用 cosine learning rate decay
param_scheduler = [
    dict(type='CosineAnnealingLR', T_max=200, by_epoch=True, begin=0, end=200)
]

# 循环设置
train_cfg = dict(type='EpochBasedTrainLoop', max_epochs=200)

../_base_/default_runtime.py 是运行时的默认配置。 运行时设置主要包含一些训练中需要使用的基础配置, 例如 default_hooks 和 log_processor

default_scope = 'mmselfsup'

default_hooks = dict(
    runtime_info=dict(type='RuntimeInfoHook'),
    optimizer=dict(type='OptimizerHook', grad_clip=None),
    timer=dict(type='IterTimerHook'),
    logger=dict(type='LoggerHook', interval=50),
    param_scheduler=dict(type='ParamSchedulerHook'),
    checkpoint=dict(type='CheckpointHook', interval=10),
    sampler_seed=dict(type='DistSamplerSeedHook'),
)

env_cfg = dict(
    cudnn_benchmark=False,
    mp_cfg=dict(mp_start_method='fork', opencv_num_threads=0),
    dist_cfg=dict(backend='nccl'),
)

log_processor = dict(
    interval=50,
    custom_keys=[dict(data_src='', method='mean', windows_size='global')])

vis_backends = [dict(type='LocalVisBackend')]
visualizer = dict(
    type='SelfSupVisualizer',
    vis_backends=vis_backends,
    name='visualizer')

log_level = 'INFO'
load_from = None
resume = False

继承和修改配置文件

为了易于理解,我们推荐贡献者从现有方法继承。

对于同一个文件夹下的所有配置,我们推荐只使用一个原始(primitive) 配置。其他所有配置应当从 原始(primitive) 配置继承。这样最大的继承层次为 3。

例如,如果你的配置文件是基于 MoCo v2 做一些修改, 首先你可以通过指定 _base_ ='./mocov2_resnet50_8xb32-coslr-200e_in1k.py.py' (相对于你的配置文件的路径)继承基本的 MoCo v2 结构,接着在配置文件中修改一些必要的参数。 现在,我们举一个更具体的例子,我们想使用 configs/selfsup/mocov2/mocov2_resnet50_8xb32-coslr-200e_in1k.py.py中几乎所有的配置,但是将训练周期数从 200 修改为 800,修改学习率衰减的时机和数据集路径,你可以创建一个名为 configs/selfsup/mocov2/mocov2_resnet50_8xb32-coslr-800e_in1k.py.py 的新配置文件,内容如下:

_base_ = './mocov2_resnet50_8xb32-coslr-200e_in1k.py'


# 学习率调整器
param_scheduler = [
    dict(type='CosineAnnealingLR', T_max=800, by_epoch=True, begin=0, end=800)
]

train_cfg = dict(type='EpochBasedTrainLoop', max_epochs=800)

使用配置中的中间变量

在配置文件中使用一些中间变量会使配置文件更加清晰和易于修改。

例如 dataset_type, train_pipeline, 是数据中的中间变量。 我们先定义它们再将它们传进 data.

# 数据集配置
# 我们使用来源于 MMClassification 中的 ``ImageNet``, 所以有一个 ``mmcls`` 的前缀
dataset_type = 'mmcls.ImageNet'
data_root = 'data/imagenet/'

# mocov2 和 mocov1 的不同主要来自于数据增强
view_pipeline = [
    dict(type='RandomResizedCrop', size=224, scale=(0.2, 1.)),
    dict(
        type='RandomApply',
        transforms=[
            dict(
                type='ColorJitter',
                brightness=0.4,
                contrast=0.4,
                saturation=0.4,
                hue=0.1)
        ],
        prob=0.8),
    dict(type='RandomGrayscale', prob=0.2, keep_channels=True),
    dict(type='RandomGaussianBlur', sigma_min=0.1, sigma_max=2.0, prob=0.5),
    dict(type='RandomFlip', prob=0.5),
]

train_pipeline = [
    dict(type='LoadImageFromFile'),
    dict(type='MultiView', num_views=2, transforms=[view_pipeline]),
    dict(type='PackSelfSupInputs', meta_keys=['img_path'])
]

train_dataloader = dict(
    batch_size=32,
    num_workers=8,
    drop_last=True,
    persistent_workers=True,
    sampler=dict(type='DefaultSampler', shuffle=True),
    collate_fn=dict(type='default_collate'),
    dataset=dict(
        type=dataset_type,
        data_root=data_root,
        ann_file='meta/train.txt',
        data_prefix=dict(img_path='train/'),
        pipeline=train_pipeline))

忽略基础配置中的字段

有时候,你需要设置 _delete_=True 来忽略基础配置文件中一些域的内容。 您可以参考 mmengine 获得更多说明。 接下来是一个例子。如果你希望在 SimCLR 使用中 MoCoV2Neck, 仅仅继承并直接修改将会报 get unexcepected keyword 'num_layers' 错误, 因为在 model.neck 域信息中,基础配置 num_layers 字段被保存下来了, 你需要添加 _delete_=True 来忽略 model.neck 在基础配置文件中的有关字段的内容:

_base_ = 'simclr_resnet50_8xb32-coslr-200e_in1k.py'

model = dict(
    neck=dict(
        _delete_=True,
        type='MoCoV2Neck',
        in_channels=2048,
        hid_channels=2048,
        out_channels=128,
        with_avg_pool=True))

使用基础配置中的字段

有时候,你可能引用 _base_ 配置中一些字段, 以避免重复定义。 你可以参考 mmengine 获取更多的说明。 下面是一个使用基础配置文件中 num_classes 的例子, 请参考 configs/selfsup/odc/odc_resnet50_8xb64-steplr-440e_in1k.py.

_base_ = [
    '../_base_/models/odc.py',
    '../_base_/datasets/imagenet_odc.py',
    '../_base_/schedules/sgd_steplr-200e_in1k.py',
    '../_base_/default_runtime.py',
]

# model settings
model = dict(
    head=dict(num_classes={{_base_.num_classes}}),
    memory_bank=dict(num_classes={{_base_.num_classes}}),
)

通过脚本参数修改配置

当用户使用脚本 “tools/train.py” 或 “tools/test.py” 提交任务,或者其他工具时,可以通过指定 --cfg-options 参数来直接修改配置文件中内容。

  • 更新字典链中的配置的键

    配置项可以通过遵循原始配置中键的层次顺序指定。例如,--cfg-options model.backbone.norm_eval=False 改变模型 backbones 中的所有 BN 模块为 train 模式。

  • 更新列表中配置的键

    你的配置中的一些配置字典是由列表组成。例如,训练 pipeline data.train.pipeline 通常是一个列表。 例如 [dict(type='LoadImageFromFile'), dict(type='TopDownRandomFlip', flip_prob=0.5), ...]。 如果你想要在 pipeline 中将 'flip_prob=0.5' 修改为 'flip_prob=0.0' , 您可以指定 --cfg-options data.train.pipeline.1.flip_prob=0.0.

  • 更新 list/tuples 中的值

    如果想要更新的值是一个列表或者元组。 例如, 一些配置文件中包含 param_scheduler = "[dict(type='CosineAnnealingLR',T_max=200,by_epoch=True,begin=0,end=200)]"。 如果你想要改变这个键,你可以指定 --cfg-options param_scheduler = "[dict(type='LinearLR',start_factor=1e-4, by_epoch=True,begin=0,end=40,convert_to_iter_based=True)]"。 注意, ” 是必要的, 并且在指定值的时候,在引号中不能存在空白字符。

导入用户定义模块

注解

这部分内容初学者可以跳过,只在使用其他 MM-codebase 时会用到,例如使用 mmcls 作为第三方库来构建你的工程。

这部分内容初学者可以跳过,只在使用其他 MM-codebase 时会用到,例如使用 mmcls 作为第三方库来构建你的工程。 i为了简化代码,你可以使用 MM-codebase 作为第三方库,只需要保存你自己额外的代码,并在配置文件中导入自定义模块。你可以参考 OpenMMLab Algorithm Competition Project 中的例子.

在你自己的配置文件中添加如下所述的代码:

custom_imports = dict(
    imports=['your_dataset_class',
             'your_transforme_class',
             'your_model_class',
             'your_module_class'],
    allow_failed_imports=False)
Read the Docs v: stable
Versions
latest
stable
1.x
dev-1.x
0.x
Downloads
pdf
html
epub
On Read the Docs
Project Home
Builds

Free document hosting provided by Read the Docs.