RoboMaster视觉教程(4)装甲板识别算法

概览

装甲板识别是RoboMaster视觉识别中比较成熟的了,到现在有很多战队开源了他们的算法。

基本上的思路都是一样的:利用装甲板灯条发光的特性将摄像头曝光值调低屏蔽环境光干扰,二值化处理图像得到只含灯条的二值图,根据装甲板灯条的几何特征来设置约束筛选灯条,灯条匹配筛选装甲板。

每年都有很多战队将他们的算法开源,善于利用他人的成果可以极大地减少自己工作量。

我17年参加比赛的时候基本上是闭门造车,一开始不知道官方已经在16年开源了一套非常好的代码。

最开始是按照车牌识别的套路来做的,自己写的代码基本上无法正确地识别装甲板。

后来折腾YOLO,终于在标记了六千多张图片训练了两天后可以识别装甲板了。识别归识别速度太慢了,那时候心情非常失落,又加上带队老师总是催进度,就弃坑学习去了。

等到学期快结束了,发现了大疆的开源代码,下下来读了读,这个真的靠谱,各个方面都想到了,甚至还有通过妙算GPIO控制LED来指示当前程序状态的代码,觉得可以做,就在考完模电后重回实验室折腾了。

换句话说那年我之前做的努力全都作废,识别装甲板、大符、小符的程序是在一个月内参考官方的(官方装甲板识别当装甲板上贴有数字就失效了)赶出来的……

下面是一些资料链接,篇篇经典!

RoboMaster论坛中总结的历届开源资料:https://bbs.robomaster.com/forum.php?mod=viewthread&tid=6979&fromuid=14

RM圆桌是这届RoboMaster推出的技术分享活动,全是干货https://www.robomaster.com/zh-CN/resource/news

RM圆桌005 抢人头要靠自瞄 https://www.robomaster.com/zh-CN/resource/pages/1009?type=newsSub

RM圆桌008 如何打击大风车 https://www.robomaster.com/zh-CN/resource/pages/1015?type=newsSub

另外一些队伍的官方公众号也会发布一些教程,例如公众号「西交RoboMaster机器人队」里有很多干货,今年大符的识别算法我就是按照他们的教程一步步做的。

「内附代码|今年的大风车能量机关识别就是这么地so easy!」https://mp.weixin.qq.com/s/3B-iR32GX7jfVyxvNQVRXw

昨天看到一句话觉得很好:一个复杂的系统并不是全部需要从0到1,把优势的资源整合在一起才能发挥最大作用,用别人的代码或思路并不是可耻的事情(要遵守对方许可证协议),站在巨人的肩膀上才可能走的更远。

今年帮这届做视觉时看到学弟在重新造轮子从零开始写装甲识别,并且也了解到去年写装甲识别的研究生也是从零开始用zed+tx2做的装甲识别最终在赛场上也没发挥作用。

我当时听了很震惊,明明我17年都解决了啊,虽然当时没有做预测没有写好基地电控导致基地自瞄很慢、没有考虑到一些意外情况在场上摄像头歪了导致打大符全打偏了,但是视觉识别的代码为什么要从头开始呢,我还花了一个星期把用到的东西算法整理了一个pdf文档,难道大家都不care前人的经验吗?

装甲板识别

由于主要参考东南大学的开源代码,他们的算法思路在readme里写的比较详细https://github.com/SEU-SuperNova-CVRA/Robomaster2018-SEU-OpenSource/tree/master/Armor

我这里讲一下我对算法做的一些改进并将整个流程过一遍。

test_infantry.cpp

在代码文件夹中的Main中有test_infantry.cpp和test_sentry.cpp,前者是步兵的程序模板,后者是哨兵的程序模板。

默认在项目文件中没有添加test_sentry.cpp,可以右键添加现存文件或者手动输入进项目文件中。这两个文件只能同时只有一个有效(即一个需要在项目文件中注释掉)。

我主要讲test_sentry.cpp,包括之后的教程都是主要用这个文件。因为test_infantry.cpp比较复杂而且其中的大符识别不需要,改起来比较麻烦,而test_sentry.cpp相对简洁,增改代码也比较方便。

在main函数里出现的三个线程中除了produce放在ImgProdCons.cpp中其他的两个都在本文件中。

void ImgProdCons::init()//各种参数的初始化
void ImgProdCons::consume()//视觉识别主程序

在识别主程序中首先定义要用到的一些变量,定义时间变量t1记录当前时间用于之后的算法耗费时间的度量,之后获取待识别图片,图片可以通过produce线程获得也可以通过视频获得,之后将图片载入到armorDetector中进行识别,识别到后的返回值在ArmorDetector.h中有定义:

enum ArmorFlag
{
    ARMOR_NO = 0,       // not found
    ARMOR_LOST = 1,     // lose tracking
    ARMOR_GLOBAL = 2,   // armor found globally
    ARMOR_LOCAL = 3     // armor found locally(in tracking mode)
};

若识别到则获取关于装甲类型、装甲板四点的图像坐标来解算装甲相对摄像头的空间坐标,之后将数据发给stm32就可以啦(发送给stm32的是拍下这张图片时以摄像头或敌方装甲为原点的相对坐标值转化成的角度值,依靠这个可以随动跟踪,但永远跟不上(─.─||)。预测比较难做我做的也不好,关于预测以后再说)。

这里可以看出来最核心的识别算法放在了ArmorDetector类中,接下来将详细讲解这部分。

分析一下装甲板

在分析代码前先来分析一下装甲板,下面这幅图是装上装甲板后的步兵轴测图(图片来自于官方RM2019裁判系统规范手册)。

carWithArmor.jpg

在图中可以看到:

  1. 装甲板竖直固定在小车的四周
  2. 同一装甲板两灯条平行、灯条长宽确定、两灯条间的间距确定
  3. 在小车转过45度后会有两个装甲板出现在摄像头画面中
  4. 两个装甲板的倾斜角度相差不大,容易将中间两个灯条误识别成大装甲板

据此可以初步构思出识别思路。

  1. 找出图片中所有的灯条
  2. 根据长宽比、面积大小和凸度来筛选灯条
  3. 对找出的灯条进行匹配找到合适的配对
  4. 配对灯条作为候选装甲板,提取其中间的图案判断是否是数字进行筛选

以上几步就是识别装甲板的步骤。

识别函数 int ArmorDetector::detect()

这个函数是算法的核心,类中的其他的各种数据结构各种成员函数都是为它服务的。

算法第一步是存储灯条对应上面分析的第一步和第二步。

我对原来的代码做了些修改,去掉了颜色识别的部分,用红蓝通道相减得到的差作为识别用的灰度图,这个方法在第一篇教程「摄像头」中提过一下。

对于图像中红色的物体来说,其rgb分量中r的值最大,g和b在理想情况下应该是0,同理蓝色物体的b分量应该最大。

如果识别红色物体可以直接用r通道-b通道。由于在低曝光下只有灯条是有颜色的,两通道相减后,其他区域的部分会因为r和b的值差不多而被减去,而蓝色灯条部分由于r通道比b通道的值小,相减后就归0了,也就是剩下的灰度图只留下了红色灯条。

// 把一个3通道图像转换成3个单通道图像
split(_roiImg,channels);//分离色彩通道
//预处理删除己方装甲板颜色
if(_enemy_color==RED)
    _grayImg=channels.at(2)-channels.at(0);//Get red-blue image;
 else _grayImg=channels.at(0)-channels.at(2);//Get blue-red image;

得到灰度图后需要阈值化处理得到二值图,之后可以进行膨胀处理让图像中的轮廓更明显。

Mat binBrightImg;
//阈值化
threshold(_grayImg, binBrightImg,_param.brightness_threshold
              , 255, cv::THRESH_BINARY);
Mat element = cv::getStructuringElement(cv::MORPH_ELLIPSE, cv::Size(3, 3));
//膨胀
dilate(binBrightImg, binBrightImg, element);

找轮廓,这步是整个算法中最耗时的部分,如果预处理做的好,可以极大地减少找轮廓中花费的时间。

vector<vector<Point>> lightContours;
//找轮廓
findContours(binBrightImg.clone(), lightContours, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_SIMPLE);

找到轮廓后开始遍历轮廓提取灯条

for(const auto& contour : lightContours)
{
          //得到面积
    float lightContourArea = contourArea(contour);
    //面积太小的不要
          if(contour.size() <= 5 ||
       lightContourArea < _param.light_min_area) continue;
    //椭圆拟合区域得到外接矩形
          RotatedRect lightRec = fitEllipse(contour);
          //矫正灯条
          adjustRec(lightRec, ANGLE_TO_UP);
          //宽高比、凸度筛选灯条
          if(lightRec.size.width / lightRec.size.height >
             _param.light_max_ratio ||
       lightContourArea / lightRec.size.area() <
             _param.light_contour_min_solidity
            )continue;
    //对灯条范围适当扩大
    lightRec.size.width *= _param.light_color_detect_extend_ratio;
    lightRec.size.height *= _param.light_color_detect_extend_ratio;
          Rect lightRect = lightRec.boundingRect();
          const Rect srcBound(Point(0, 0), _roiImg.size());
          lightRect &= srcBound;
          //因为颜色通道相减后己方灯条直接过滤,不需要判断颜色了,可以直接将灯条保存
          lightInfos.push_back(LightDescriptor(lightRec));
      }
//没找到灯条就返回没找到
if(lightInfos.empty())
{
    return _flag = ARMOR_NO;
}

对灯条进行匹配筛选

      //按灯条中心x从小到大排序
sort(lightInfos.begin(), lightInfos.end(), [](const LightDescriptor& ld1, const LightDescriptor& ld2)
      {//Lambda函数,作为sort的cmp函数
    return ld1.center.x < ld2.center.x;
});
for(size_t i = 0; i < lightInfos.size(); i++)
{//遍历所有灯条进行匹配
    for(size_t j = i + 1; (j < lightInfos.size()); j++)
          {
        const LightDescriptor& leftLight  = lightInfos[i];
        const LightDescriptor& rightLight = lightInfos[j];
        /*
        *   Works for 2-3 meters situation
        *   morphologically similar: // parallel 
                         // similar height
        */
              //角差
              float angleDiff_ = abs(leftLight.angle - rightLight.angle);
              //长度差比率
              float LenDiff_ratio = abs(leftLight.length - rightLight.length) / max(leftLight.length, rightLight.length);
              //筛选
              if(angleDiff_ > _param.light_max_angle_diff_ ||
           LenDiff_ratio > _param.light_max_height_diff_ratio_)
        {
            continue;
        }

        /*
        *   proper location:  y value of light bar close enough 
        *             ratio of length and width is proper
        */
              //左右灯条相距距离
        float dis = cvex::distance(leftLight.center, rightLight.center);
              //左右灯条长度的平均值
              float meanLen = (leftLight.length + rightLight.length) / 2;
              //左右灯条中心点y的差值
              float yDiff = abs(leftLight.center.y - rightLight.center.y);
              //y差比率
              float yDiff_ratio = yDiff / meanLen;
              //左右灯条中心点x的差值
              float xDiff = abs(leftLight.center.x - rightLight.center.x);
              //x差比率
              float xDiff_ratio = xDiff / meanLen;
              //相距距离与灯条长度比值
              float ratio = dis / meanLen;
              //筛选
              if(yDiff_ratio > _param.light_max_y_diff_ratio_ ||
           xDiff_ratio < _param.light_min_x_diff_ratio_ ||
           ratio > _param.armor_max_aspect_ratio_ ||
           ratio < _param.armor_min_aspect_ratio_)
        {
            continue;
        }

        // calculate pairs' info 
              //按比值来确定大小装甲
        int armorType = ratio > _param.armor_big_armor_ratio ? BIG_ARMOR : SMALL_ARMOR;
        // calculate the rotation score
        float ratiOff = (armorType == BIG_ARMOR) ? max(_param.armor_big_armor_ratio - ratio, float(0)) : max(_param.armor_small_armor_ratio - ratio, float(0));
        float yOff = yDiff / meanLen;
        float rotationScore = -(ratiOff * ratiOff + yOff * yOff);
              //得到匹配的装甲板
              ArmorDescriptor armor(leftLight, rightLight, armorType, channels.at(1), rotationScore, _param);

        _armors.emplace_back(armor);
        break;
    }
}
//没匹配到装甲板则返回没找到
if(_armors.empty())
{
    return _flag = ARMOR_NO;
}

对找到的装甲板进行筛选

//delete the fake armors
   _armors.erase(remove_if(_armors.begin(), _armors.end(), [this](ArmorDescriptor& i)
   {//lamdba函数判断是不是装甲板,将装甲板中心的图片提取后让识别函数去识别,识别可以用svm或者模板匹配等
       return 0==(i.isArmorPattern(_small_Armor_template,_big_Armor_template,lastEnemy));
}), _armors.end());
//全都判断不是装甲板
if(_armors.empty())
{
    _targetArmor.clear();

    if(_flag == ARMOR_LOCAL)
    {
        //cout << "Tracking lost" << endl;
        return _flag = ARMOR_LOST;
    }
    else
    {
        //cout << "No armor pattern detected." << endl;
        return _flag = ARMOR_NO;
    }
}

判断是不是装甲板我用的是模板匹配的方法,模板匹配特别适合待识别图片不会变化的场景,选择合适的模板可以得到很高的准确率并且花费时间远小于svm等机器学习方法。

//模板匹配 根据装甲板中心的图案判断是不是装甲板
bool ArmorDescriptor::isArmorPattern(std::vector<cv::Mat> &small,
                                     std::vector<cv::Mat> &big ,
                                     LastenemyType &lastEnemy)
{
    //若需要判断装甲中间数字
#ifdef IS_ARMOR
    vector<pair<int,double>> score;
    map<int,double> mp;
    Mat regulatedImg=frontImg;
    
    for(int i=0;i<8;i++){
        //载入模板,模板是在初始化的时候载入ArmorDetector类,
        //因为ArmorDescriptor与其非同类需要间接导入
        Mat tepl=small[i];
        Mat tepl1=big[i];
        //模板匹配得到位置,这里没用
        cv::Point matchLoc;
        //模板匹配得分
        double value;
        //匹配小装甲
        value = TemplateMatch(regulatedImg, tepl, matchLoc, CV_TM_CCOEFF_NORMED);
        mp[i+1]=value;
        score.push_back(make_pair(i+1,value));
        //匹配大装甲
        value = TemplateMatch(regulatedImg, tepl1, matchLoc, CV_TM_CCOEFF_NORMED);
        mp[i+11]=value;
        score.push_back(make_pair(i+11,value));
    }
    //对该装甲与所有模板匹配后的得分进行排序
    sort(score.begin(),score.end(), [](const pair<int,double> &a, const pair<int,double> &b)
    {
        return a.second > b.second;
    });
    //装甲中心位置
    cv::Point2f c=(vertex[0]+vertex[1]+vertex[2]+vertex[3])/4;
    //装甲数字即为得分最高的那个
    int resultNum=score[0].first;
    //得分太低认为没识别到数字
    if(score[0].second<0.6)
    {
        if(//与上次识别到的装甲板位置差不多,且丢失次数不超过一定值
                std::abs(std::abs(lastEnemy.center.x)-std::abs(c.x))<10&&
                std::abs(std::abs(lastEnemy.center.y)-std::abs(c.y))<10&&
                lastEnemy.lostTimes<100
                )
        {//认为该装甲的数字与上次相同
            lastEnemy.lostTimes++;
            lastEnemy.center=c;
            enemy_num=lastEnemy.num;
            return true;
        }
        else
        {//认为不是装甲
            return false;
        }
    }
    //当装甲板识别为小装甲,而得到的号码为11、22……时,说明把大装甲识别为了小装甲。
    if(type==SMALL_ARMOR )
    {
        if(resultNum>10)
        {
            type=BIG_ARMOR;
        }
    }
    enemy_num=resultNum%10;
    lastEnemy.num=enemy_num;
    lastEnemy.center=c;
    lastEnemy.lostTimes=0;
    return true;
#endif
    //若不需要判断匹配的装甲板中的数字则整个函数直接返回true
#ifndef IS_ARMOR
    return true;
#endif
}

对装甲板筛选后可能会有多个装甲板,这时候需要选定一个进行跟踪和打击

   //找出历史装甲中次数最多装甲号
   int targetNum=0;
   if(!enemy_nums.empty())
   {
       //一共8个装甲号
       int count[9]={0};
       for(auto num:enemy_nums)
          count[num]++;
       // 找出最多出现的次数
       int maxCount=0;
       for(int i = 1; i < 9; i++)  
       {
           if(count[i] > maxCount)
               maxCount = count[i];
       }
       // 找出出现最多次的那个数字
       for(int i = 1; i < 9; i++)            
       {
           if(count[i] == maxCount)
               targetNum = i;
       }
       //保留最近的10个装甲号
       if(enemy_nums.size()>10)
       {
           enemy_nums.erase(enemy_nums.begin());
       }
   }
   //选择装甲板
   bool findFlag=false;
   if(targetNum!=0)
   {
       for(auto & armor : _armors)
       {
           //跟踪之前出现的那辆车的装甲板
           if(armor.enemy_num==targetNum)
           {
               findFlag=true;
               _targetArmor=armor;
               break;
           }
       }
   }
//之前没有跟踪或者之前的车不见了
   if(findFlag==false)
   {
       //calculate the final score
       for(auto & armor : _armors)
       {
           armor.finalScore = armor.sizeScore + armor.distScore + armor.rotationScore;
       }

       //choose the one with highest score, store it on _targetArmor
       std::sort(_armors.begin(), _armors.end(), [](const ArmorDescriptor & a, const ArmorDescriptor & b)
       {
           return a.finalScore > b.finalScore;
       });
       //选择得分最高的目标装甲板
       _targetArmor = _armors[0];
   }
//update the flag status    
_trackCnt++;

   enemy_nums.push_back(_targetArmor.enemy_num);
return _flag = ARMOR_LOCAL;

装甲识别算法讲完了!觉得不错点个赞呗(^_^)

申请了一个自己的公众号 江达小记 ,打算将自己的学习研究的经验总结下来帮助他人也方便自己。感兴趣的朋友可以关注一下。

qrcode_for_gh_49ee48ab432a_258.jpg

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

推荐阅读更多精彩内容