基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

本文目录

简述

MMRotate 是一款基于 PyTorch 的旋转框检测的开源工具箱,是 OpenMMLab 项目的成员之一。里面包含了rcnn、faster rcnn、r3det等各种旋转目标的检测模型,适合于遥感图像领域的目标检测。

1.MMrotate下载

MMrotate包下载: 下载链接
目录结构如下

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

; 2.环境安装

所需要的依赖环境

    Linux & Windows
    Python 3.7+
    PyTorch 1.6+
    CUDA 9.2+
    GCC 5+
    mmcv-full 1.4.5+
    mmdet 2.19.0+

安装流程:


conda create -n mmrotate python=3.7 -y
conda activate mmrotate

conda install pytorch==1.7.0 torchvision==0.8.0 cudatoolkit=10.1 -c pytorch

pip install mmcv-full -f https://download.openmmlab.com/mmcv/dist/cu101/torch1.7.0/index.html

pip install mmdet

cd mmrotate
pip install -r requirements/build.txt
pip install -v -e .

测试安装的环境是否正常:
打开demo文件夹下的image_demo.py:修改相应的参数:

– -img : 测试的图片
– -config:选择测试的模型配置文件
– -checkpoint: 对应的预训练模型文件,可自行下载:MMrotate模型库 百度网盘连接:rcnn 权重 提取码:ur0b
– -output: 检测输出的图片

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30
然后,执行并得到如下结果,说明环境安装正确:
python demo/image_demo.py

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

3.自定义数据集制作

要训练自己的模型,自定义数据集制作这部分其实是最麻烦的。MMrotate所使用的数据集格式是dota类型的,图片为.png格式且尺寸是 n×n 的(方形),不过不用担心,该项目中有相应的工具包可自动转换
DOTA的标签格式

x1, y1, x2, y2, x3, y3, x4, y4, category: 目标名字 difficult:表示标签检测的难易程度 (1表示困难,0表示不困难)

x1,y1为左上角的坐标,然后顺时针排列4个坐标

3.1 roLabelImg 打标签

roLabelImg工具算是labelme的进阶版吧,它能把矩形框进行旋转,制作旋转目标的标签。操作时:A键 D键 Z键 X键 C键 V键 这几个常用快捷键用得多点
生成的xml标签文件如下

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

其中相比普通的标签文件多了一个角度信息(angle),后续转为DOTA的标签时就是利用了该角度(这里强烈建议先看大佬的文章:旋转框的转换问题;),所以在用roLabelImg打标签时一定要注意三点:(1)roLabelImg中本身的角度是以框的w边 与x轴顺时针之间所形成的夹角为正角(2)旋转框的w边(初始水平边)始终旋转到与目标的长边重合(3)打标签时保证angle在[-90°,90°]间,。这种标注法原项目叫做”长边标注法”(le90),具体见下图:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

至此,生成的目录结构如下:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

3.2 生成DOTA数据集格式的标签

利用上一步中xml标签 转换为DOTA的标签,转换的代码(只需要给定3个路径参数就可以运行了):


import os
import xml.etree.ElementTree as ET
import math
import cv2 as cv

def voc_to_dota(xml_dir, xml_name, img_dir, savedImg_dir):
    txt_name = xml_name[:-4] + '.txt'
    txt_path = xml_dir + '/txt_label'
    if not os.path.exists(txt_path):
        os.makedirs(txt_path)
    txt_file = os.path.join(txt_path, txt_name)

    img_name = xml_name[:-4] + '.jpg'
    img_path = os.path.join(img_dir, img_name)
    img = cv.imread(img_path)

    xml_file = os.path.join(xml_dir, xml_name)
    tree = ET.parse(os.path.join(xml_file))
    root = tree.getroot()
    with open(txt_file, "w+", encoding='UTF-8') as out_file:

        for obj in root.findall('object'):
            name = obj.find('name').text
            difficult = obj.find('difficult').text

            robndbox = obj.find('robndbox')
            cx = float(robndbox.find('cx').text)
            cy = float(robndbox.find('cy').text)
            w = float(robndbox.find('w').text)
            h = float(robndbox.find('h').text)
            angle = float(robndbox.find('angle').text)

            p0x, p0y = rotatePoint(cx, cy, cx - w / 2, cy - h / 2, -angle)
            p1x, p1y = rotatePoint(cx, cy, cx + w / 2, cy - h / 2, -angle)
            p2x, p2y = rotatePoint(cx, cy, cx + w / 2, cy + h / 2, -angle)
            p3x, p3y = rotatePoint(cx, cy, cx - w / 2, cy + h / 2, -angle)

            dict = {p0y:p0x, p1y:p1x, p2y:p2x, p3y:p3x }
            list = find_topLeftPopint(dict)

            if list[0] == p0x:
                list_xy = [p0x, p0y, p1x, p1y, p2x, p2y, p3x, p3y]
            elif list[0] == p1x:
                list_xy = [p1x, p1y, p2x, p2y, p3x, p3y, p0x, p0y]
            elif list[0] == p2x:
                list_xy = [p2x, p2y, p3x, p3y, p0x, p0y, p1x, p1y]
            else:
                list_xy = [p3x, p3y, p0x, p0y, p1x, p1y, p2x, p2y]

            cv.line(img, (int(list_xy[0]), int(list_xy[1])), (int(list_xy[2]), int(list_xy[3])), color=(255, 0, 0), thickness= 3)
            cv.line(img, (int(list_xy[2]), int(list_xy[3])), (int(list_xy[4]), int(list_xy[5])), color=(0, 255, 0), thickness= 3)
            cv.line(img, (int(list_xy[4]), int(list_xy[5])), (int(list_xy[6]), int(list_xy[7])), color=(0, 0, 255), thickness= 2)
            cv.line(img, (int(list_xy[6]), int(list_xy[7])), (int(list_xy[0]), int(list_xy[1])), color=(255, 255, 0), thickness= 2)

            data = str(list_xy[0]) + " " + str(list_xy[1]) + " " + str(list_xy[2]) + " " + str(list_xy[3]) + " " + \
                   str(list_xy[4]) + " " + str(list_xy[5]) + " " + str(list_xy[6]) + " " + str(list_xy[7]) + " "
            data = data + name + " " + difficult + "\n"
            out_file.write(data)
        if not os.path.exists(savedImg_dir):
            os.makedirs(savedImg_dir)
        out_img = os.path.join(savedImg_dir, xml_name[:-4]+'.jpg')
        cv.imwrite(out_img, img)

def find_topLeftPopint(dict):
    dict_keys = sorted(dict.keys())
    temp = [dict[dict_keys[0]], dict[dict_keys[1]]]
    minx = min(temp)
    if minx == temp[0]:
        miny = dict_keys[0]
    else:
        miny = dict_keys[1]
    return [minx, miny]

def rotatePoint(xc, yc, xp, yp, theta):
    xoff = xp - xc
    yoff = yp - yc
    cosTheta = math.cos(theta)
    sinTheta = math.sin(theta)
    pResx = cosTheta * xoff + sinTheta * yoff
    pResy = - sinTheta * xoff + cosTheta * yoff

    return float(format(xc + pResx, '.1f')), float(format(yc + pResy, '.1f'))

import argparse
def parse_args():
    parser = argparse.ArgumentParser(description='数据格式转换')
    parser.add_argument('--xml-dir', default='./buildings/train/label', help='original xml file dictionary')
    parser.add_argument('--img-dir', default='./buildings/train', help='original image dictionary')
    parser.add_argument('--outputImg-dir', default='./buildings/train/out', help='saved image dictionary after dealing ')

    args = parser.parse_args()
    return args

if __name__ == '__main__':
    args = parse_args()
    xml_path = args.xml_dir
    xmlFile_list = os.listdir(xml_path)
    print(xmlFile_list)
    for i in range(0, len(xmlFile_list)):
        if ('.xml' in xmlFile_list[i]) or ('.XML' in xmlFile_list[i]):
            voc_to_dota(xml_path, xmlFile_list[i], args.img_dir, args.outputImg_dir)
            print('----------------------------------------{}{}----------------------------------------'
                  .format(xmlFile_list[i], ' has Done!'))
        else:
            print(xmlFile_list[i] + ' is not xml file')

其中转换的原理就是基于大佬的知乎文章中转换矩阵,生成的DOTA标签格式如下:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30
目前所得到文件夹结构如下
基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

images: 存放的.jpg图像(我只有200张)
xml: rolabelImg打的标签(这时已没啥用了,可删除)
txt_Label: 生成的DOTA标签

然后将上述数据集分成 train、test、val 、trainval等几部分以便于训练,其中每个部分的文件夹下都包含有 images(图像) 和 labels(对应的txt标签)
我这里是手动划分的:train 80%,test 10%,val 10%。

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

3.3 数据集裁剪(split)

然后将得到的 train、test、val 中的图片进行裁剪,裁剪的原因 是由于我的图像是1920×1080的矩形,而且也不是png格式的,所以找到:mmrotate-main/tools/data/dota/split/ 路径下img_split.py文件(裁剪脚本) 以及 mmrotate-main/tools/data/dota/split/split_configs/ 路径下的配置文件,其文件内容就是img_split.py的配置信息,我们需要修改其中的参数,让其加载上述的train、test、val中的图像及标签,并进行裁剪。

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30
修改好上述几个关键的参数之后,即可运行img_split.py进行裁剪了:
python mmrotate-main/tools/data/dota/split/img_split.py --base_json mmrotate-main/tools/data/dota/split/split_configs/ss_train.json

其他 test、val、trainval数据集的裁剪同理,只需修改 – -base_json参数为对应的配置文件即可。
裁剪的结果:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

4.修改配置文件

数据集准备好之后,接下来,需要修改训练相关的配置信息:主要就是 tools/tain.py 中的相关参数修改,train.py文件如下:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

两个主要的参数: – -config: 使用的模型文件 (我使用的是 faster rcnn) ; – -work-dir:训练得到的模型及配置信息保存的路径

由于使用的模型是:rotated_faster_rcnn_r50_fpn_x1_dota_le90.py,在该文件中实现了:训练相关参数的配置,数据集信息的配置,以及faster-rcnn网络架构的搭建;打开该文件修改其中的目标类别数为自己数据集的类别数,我的类别为1, 所以修改:num_classes = 1,

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

同时,修改 mmrotate-main/mmrotate/datasets/dota.py 文件中的类别名字(CLASSES), 具体如下:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

最后,修改训练使用的数据集路径:找到并打开 mmrotate-main/configs/ base/datasets/dotav1.py 文件,修改其中的 data_root 路径为自己裁剪的数据集路径,该路径包含了上述划分的train、test、val、trainval数据集,另外,dotav1.py文件中data下的train、val、test三个对象分别就是利用上述路径下对应类别的数据集,网络加载数据也是通过这三个对象中的路径去加载的,所以我们自己划分的数据集实质上只用到了:train数据集、trainval数据集、test数据集。(因为我有其他训练任务所以划分了4个),综上所述,数据集的划分完全可以根据这三个对象路径来划分就行,并不一定要和本文的一样,保证数据能正常加载训练即可。

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

; 5.训练并测试

一切准备就绪后 即可开始愉快的训练了!!!执行:

python tools/train.py

训练结果:

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30
保存训练的结果在 run 目录下:
基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

1 学习率,batch,num-worker,epoch,优化函数等 都是在configs/ base/下的文件内去修改
2 如果遇到 Cuda out of memory错误:可将 mmrotate-main/configs/ base/datasets/dotav1.py文件中的samples_per_gpu 和 workers_per_gpu 改小一点。

基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30
测试效果:
基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

6 公开数据集及对应代码下载网址

比较全面的数据集网站: 根据关键词搜索基本上都能找到相关的数据集

参考博客

AI目标检测:VOC格式数据集转换为DOTA类型数据集

使用MMrotate的Kfiou进行身份证分割

Original: https://blog.csdn.net/qq_43581224/article/details/123838415
Author: YD-阿三
Title: 基于MMRotate训练自定义数据集 做旋转目标检测 2022-3-30

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/669409/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • 3D视觉(三):双目摄像头的标定与校正

    3D视觉(三):双目摄像头的标定与校正 对于双目摄像头而言,除了需要分别标定左目摄像头的内参矩阵K1、畸变系数D1、右目摄像头的内参矩阵K2、畸变系数D2,还需要标定左右目对应的旋…

    人工智能 2023年5月26日
    052
  • unbuntu20.04下载opencv4.5.2

    最近因为某种原因不小心把电脑格式化了(啥也别说,就是一个大写的惨啊,呜呜呜呜,而且还没有备份文件)所以这里提醒大家,文件一定要备份!一定!一定!一定。重装系统之后,现在下载回各种软…

    人工智能 2023年7月18日
    050
  • 特斯拉陷入巨大质疑:车祸前1秒,Autopilot自动退出

    金磊 丰色 发自 凹非寺 量子位 | 公众号 QbitAI 一场针对 特斯拉的调查,在社交媒体上掀起了 轩然大波。 甚至有网友用 “史诗级”来形容这次调查的…

    人工智能 2023年6月11日
    062
  • 综述-自动驾驶中基于图像的3D目标检测

    更新日志: 22.03.29 完成对论文的粗读22.04.03 根据综述分类线索,查找对应论文阅读并记录了各自亮点22.04.07 着手对综述论文和提及方法(3DOP)的解析与记录…

    人工智能 2023年7月10日
    0107
  • 深度学习系列45:图像恢复综述

    从本期开始,会探索图像恢复领域的论文和代码。本次先阅读一下综述。传统方法一个很大的假设是我们相信我们可以在缺失区域之外找到相似的patch,但是如果缺失区域之外没有任何类似的pat…

    人工智能 2023年6月24日
    092
  • Ubuntu18.04+Android手机IMU+ROS Melodic跑ORB-SLAM2

    Ubuntu18.04+Android手机IMU+ROS Melodic跑ORB-SLAM2 前言 一、ROS Melodic在ubuntu系统18.04版本上的安装 二、基于RO…

    人工智能 2023年6月16日
    093
  • Blockchain is Watching You: Profiling and Deanonymizing Ethereum Users

    今天给大家讲解的论文是关于构建区块链用户画像的,它的中文题目是《区块链正在注视着你:对以太坊用户进行分析和去匿名化》 文章目录 相关概念 * 准标识符 Quasi-identifi…

    人工智能 2023年7月16日
    090
  • MySQL-索引

    一、介绍 索引是数据库对象之一,用于提高字段检索效率,使用者只需要对哪个表中哪些字段建立索引即可,其余什么都不做,数据库会自行处理。 索引提供指向存储在表的指定列中的数据值的指针,…

    人工智能 2023年7月30日
    086
  • 卷积层中的空间金字塔池化(Spatia

    详细解决卷积层中的空间金字塔池化 在卷积神经网络(Convolutional Neural Network,简称CNN)中,空间金字塔池化(Spatial Pyramid Pool…

    人工智能 2024年1月1日
    062
  • 红外图像和热成像图像

    一、简单介绍 红外摄像头和红外热成像是两个完全不一样的东西。两种相机的核心区别主要在于传感器捕捉的光波长不同。 普通相机加装滤光片可得到红外图像,原因在于普通相机的光学传感器本身覆…

    人工智能 2023年6月17日
    093
  • 记录解决cmd打不开/闪退

    1.起因 : 事情要从万恶的环境配置说起。作为一个受anaconda折磨多次的资深bug maker(bushi),这次我又又又又又又又碰到问题了。 由于一些深度学习包配置问题,我…

    人工智能 2023年7月26日
    088
  • 自然语言处理(十):LSTM模型

    LSTM(Long Short-Term Memory)也称长短时记忆结构,它是传统RNN的变体, 与经典RNN相比能够有效捕捉长序列之间的语义关联,缓解梯度消失或爆炸现象,同时L…

    人工智能 2023年5月31日
    091
  • oppo问答系统技术路线

    本笔记主要解析DataFunTalk公众号上发布的文章《李向林:OPPO自研大规模知识图谱及其在小布助手中的应用》中问答系统搭建方面的内容 文章目录 1、领域分类 2、结构化问答 …

    人工智能 2023年6月1日
    0109
  • 数字图像处理第六章——彩色图像处理

    目录 引言 一、彩色基础 二、彩色模型 2.1 RGB彩色模型 2.2 CMY和CMYK彩色模型 2.3 HSI彩色模型 三、伪彩色图像处理 3.1 灰度分层 3.2 灰度到彩色的…

    人工智能 2023年7月28日
    072
  • values)

    问题描述 在数据分析和机器学习任务中,我们经常需要对数据进行评估和排序,以便找到最相关或最重要的样本。其中一个常见的方法是使用价值估计(values estimation)的技术。…

    人工智能 2024年1月1日
    049
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球