# 自定义数据集
在本教程中,我们将介绍如何通过在线转换来自定义你的数据集。
- [自定义数据集](#自定义数据集)
- [MMAction2 数据集概述](#mmaction2-数据集概述)
- [定制新的数据集](#定制新的数据集)
- [为 PoseDataset 自定义关键点格式](#为-posedataset-自定义关键点格式)
## MMAction2 数据集概述
MMAction2 提供了任务特定的 `Dataset` 类,例如用于动作识别的 `VideoDataset`/`RawframeDataset`,用于时空动作检测的 `AVADataset`,用于基于骨骼的动作识别的`PoseDataset`。这些任务特定的数据集只需要实现 `load_data_list(self)` 来从注释文件生成数据列表。剩下的函数由超类(即 `BaseActionDataset` 和 `BaseDataset`)自动处理。下表显示了模块的继承关系和主要方法。
| 类名 | 类方法 |
| ------------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `MMAction2::VideoDataset` | `load_data_list(self)`
从注释文件中构建数据列表。 |
| `MMAction2::BaseActionDataset` | `get_data_info(self, idx)`
给定 `idx`,从数据列表中返回相应的数据样本。 |
| `MMEngine::BaseDataset` | `__getitem__(self, idx)`
给定 `idx`,调用 `get_data_info` 获取数据样本,然后调用 `pipeline` 在 `train_pipeline` 或 `val_pipeline` 中执行数据变换和增强。 |
## 定制新的数据集类
大多数情况下,把你的数据集离线转换成指定格式是首选方法,但 MMAction2 提供了一个方便的过程来创建一个定制的 `Dataset` 类。如前所述,任务特定的数据集只需要实现 `load_data_list(self)` 来从注释文件生成数据列表。请注意,`data_list` 中的元素是包含后续流程中必要字段的 `dict`。
以 `VideoDataset` 为例,`train_pipeline`/`val_pipeline` 在 `DecordInit` 中需要 `'filename'`,在 `PackActionInputs` 中需要 `'label'`。因此,`data_list` 中的数据样本必须包含2个字段:`'filename'`和`'label'`。
请参考[定制数据流水线](customize_pipeline.md)以获取有关 `pipeline` 的更多详细信息。
```
data_list.append(dict(filename=filename, label=label))
```
`AVADataset` 会更加复杂,`data_list` 中的数据样本包含有关视频数据的几个字段。此外,它重写了 `get_data_info(self, idx)` 以转换在时空动作检测数据流水线中需要用的字段。
```python
class AVADataset(BaseActionDataset):
...
def load_data_list(self) -> List[dict]:
...
video_info = dict(
frame_dir=frame_dir,
video_id=video_id,
timestamp=int(timestamp),
img_key=img_key,
shot_info=shot_info,
fps=self._FPS,
ann=ann)
data_list.append(video_info)
data_list.append(video_info)
return data_list
def get_data_info(self, idx: int) -> dict:
...
ann = data_info.pop('ann')
data_info['gt_bboxes'] = ann['gt_bboxes']
data_info['gt_labels'] = ann['gt_labels']
data_info['entity_ids'] = ann['entity_ids']
return data_info
```
## 为 PoseDataset 自定义关键点格式
MMAction2 目前支持三种关键点格式:`coco`,`nturgb+d` 和 `openpose`。如果你使用其中一种格式,你可以简单地在以下模块中指定相应的格式:
对于图卷积网络,如 AAGCN,STGCN,...
- `pipeline`:在 `JointToBone` 中的参数 `dataset`。
- `backbone`:在图卷积网络中的参数 `graph_cfg`。
对于 PoseC3D:
- `pipeline`:在 `Flip` 中,根据关键点的对称关系指定 `left_kp` 和 `right_kp`。
- `pipeline`:在 `GeneratePoseTarget` 中,如果 `with_limb` 为 `True`,指定`skeletons`,`left_limb`,`right_limb`,如果 `with_kp` 为 `True`,指定`left_kp` 和 `right_kp`。
如果使用自定义关键点格式,需要在 `backbone` 和 `pipeline` 中都包含一个新的图布局。这个布局将定义关键点及其连接关系。
以 `coco` 数据集为例,我们在 `Graph` 中定义了一个名为 `coco` 的布局。这个布局的 `inward` 连接包括所有节点连接,每个**向心**连接由一个节点元组组成。`coco`的额外设置包括将节点数指定为 `17`,将 `node 0` 设为中心节点。
```python
self.num_node = 17
self.inward = [(15, 13), (13, 11), (16, 14), (14, 12), (11, 5),
(12, 6), (9, 7), (7, 5), (10, 8), (8, 6), (5, 0),
(6, 0), (1, 0), (3, 1), (2, 0), (4, 2)]
self.center = 0
```
同样,我们在 `JointToBone` 中定义了 `pairs`,添加了一个 bone `(0, 0)` 以使 bone 的数量对齐到 joint。coco数据集的 `pairs` 如下所示,`JointToBone` 中的 `pairs` 的顺序无关紧要。
```python
self.pairs = ((0, 0), (1, 0), (2, 0), (3, 1), (4, 2),
(5, 0), (6, 0), (7, 5), (8, 6), (9, 7),
(10, 8), (11, 0), (12, 0), (13, 11), (14, 12),
(15, 13), (16, 14))
```
要使用你的自定义关键点格式,只需定义上述设置为你的图结构,并在你的配置文件中指定它们,如下所示。在这个例子中,我们将使用 `STGCN`,其中 `n` 表示类别的数量,`custom_dataset` 在 `Graph` 和 `JointToBone` 中定义。
```python
model = dict(
type='RecognizerGCN',
backbone=dict(
type='STGCN', graph_cfg=dict(layout='custom_dataset', mode='stgcn_spatial')),
cls_head=dict(type='GCNHead', num_classes=n, in_channels=256))
train_pipeline = [
...
dict(type='GenSkeFeat', dataset='custom_dataset'),
...]
val_pipeline = [
...
dict(type='GenSkeFeat', dataset='custom_dataset'),
...]
test_pipeline = [
...
dict(type='GenSkeFeat', dataset='custom_dataset'),
...]
```
只需简单地指定自定义布局,你就可以使用你自己的关键点格式进行训练和测试了。通过这种方式,MMAction2 为用户提供了很大的灵活性,允许用户自定义他们的数据集和关键点格式,以满足他们特定的需求。
以上就是关于如何自定义你的数据集的一些方法。希望这个教程能帮助你理解MMAction2的数据集结构,并教给你如何根据自己的需求创建新的数据集。虽然这可能需要一些编程知识,但是 MMAction2 试图使这个过程尽可能简单。通过了解这些基本概念,你将能够更好地控制你的数据,从而改进你的模型性能。