opencv——感兴趣区域(ROI)的分析和选取[详细总结]

引言

在利用OpenCV对图像进行处理时,通常会遇到一个情况,就是只需要对部分感兴趣区域进行处理。因此,如何选取感兴趣区域呢?(其实就是”抠图”)。

在学习opencv的掩码运算后,尝试实现一个类似halcon的reduce_domain功能,对于实现抠图的过程中,需要掌握的要点就是位运算符和copyTo函数👏

  • *位运算符的相关API:
void bitwise_and(InputArray src1, InputArray src2, OutputArray dst);  //dst = src1 & src2  “与”操作
void bitwise_or(InputArray src1, InputArray src2, OutputArray dst);  //dst = src1 | src2   “或”操作
void bitwise_xor(InputArray src1, InputArray src2, OutputArray dst);   //dst = src1 ^ src2 “异或”操作
void bitwise_not(InputArray src, OutputArray dst);  //dst = ~src                           “非”操作
  • *copyTo函数它的定义

OpenCV中image.copyTo()有两种形式:

1、image.copyTo(imageROI),作用是把image的内容复制到imageROI;

2、image.copyTo(imageROI,mask),作用是把原图(image)和掩膜(mask)与运算后得到ROI区域(imageROI)。

opencv——感兴趣区域(ROI)的分析和选取[详细总结]

mask就是位图,如果mask像素的值是非0的,我就拷贝它,否则不拷贝。(非零的位置就是原图中的那些需要拷贝的部分)

正文部分

对于感兴趣区域(Region of Interest, ROI)的选取,一般有两种情形:1)已知ROI在图像中的位置;2)ROI在图像中的位置未知。

1)第一种情形很简单,根据ROI的坐标直接从原图抠出,不过前提是要知道其坐标,

  • *矩形ROI区域提取(将ROI存放于一张新的图像里)
int main(int argc, char** argv)
{
    Mat src, mask,dst;
    Rect r1(80, 80, 200, 200);//创建矩形ROI区域
    src = imread("D:/opencv练习图片/薛之谦.jpg");
    mask = Mat::zeros(src.size(), CV_8UC1);//创建纯黑色二值图像
    mask(r1).setTo(255);//构建掩膜(将矩形ROI涂白)
    src.copyTo(dst, mask);//掩膜运算
    imshow("ROI区域", dst);
    imshow("掩膜", mask    );
    waitKey(0);
    return 0;
}

🙄注意程序中的这两句关于Mask的操作。

mask = Mat::zeros(src.size(), CV_8UC1);
mask(r1).setTo(255);  //r1是设置好的感兴趣区域

解释一下上面两句的操作。

  1. 第一步建立与原图一样大小的mask图像,并将所有像素初始化为0,因此全图成了一张全黑色的二值图。
  2. 第二步将mask图中的r1区域的所有像素值设置为255,也就是整个r1区域变成了白色。

opencv——感兴趣区域(ROI)的分析和选取[详细总结]

若要只是提取ROI区域,这一句代码(两种书写方法)即可

    // 指定感兴趣区域,两种书写方法
    Mat ROI = src(Rect(80, 80, 200, 200));
    Mat ROI2(src, Rect(80, 80, 200, 200));

opencv——感兴趣区域(ROI)的分析和选取[详细总结]
  • *但是若要提取不规则区域ROI,该怎么处理呢?

答案:使用 contour (轮廓)来指定ROI区域

int main(int argc, char** argv)
{
    Mat src,dst;
    src = imread("D:/opencv练习图片/薛之谦.jpg");
    Mat ROI = Mat::zeros(src.size(), CV_8UC1);
    vector<vector<point>> contours;//&#x8F6E;&#x5ED3;
    vector<point> pts;//&#x591A;&#x8FB9;&#x5F62;&#x89D2;&#x70B9;&#x96C6;&#x5408;
    pts.push_back(Point(30, 45));
    pts.push_back(Point(100, 15));//push_back() &#x5728;Vector&#x6700;&#x540E;&#x6DFB;&#x52A0;&#x4E00;&#x4E2A;&#x5143;&#x7D20;(&#x53C2;&#x6570;&#x4E3A;&#x8981;&#x63D2;&#x5165;&#x7684;&#x503C;)
    pts.push_back(Point(200, 145));
    pts.push_back(Point(300, 240));
    pts.push_back(Point(50, 250));
    contours.push_back(pts);
    drawContours(ROI, contours, 0, Scalar(255), -1);//&#x7528;&#x767D;&#x8272;&#x586B;&#x5145;&#x591A;&#x8FB9;&#x5F62;&#x533A;&#x57DF;
    src.copyTo(dst, ROI);//&#x63A9;&#x7801;&#x8FD0;&#x7B97;
    imshow("&#x63A9;&#x7801;", ROI);
    imshow("img", src);
    imshow("dst", dst);
    waitKey(0);
    return 0;
}</point></vector<point>

opencv——感兴趣区域(ROI)的分析和选取[详细总结]
  • *若是只想要一个圆形区域呢?
int main(int argc, char** argv)
{
    Mat src;
    src = imread("D:/opencv&#x7EC3;&#x4E60;&#x56FE;&#x7247;/&#x859B;&#x4E4B;&#x8C26;.jpg");
    Mat dst = Mat::zeros(src.size(), src.type());//&#x521B;&#x5EFA;&#x4E09;&#x901A;&#x9053;&#x56FE;&#x50CF;
    Mat mask = Mat::zeros(src.size(), CV_8U);//&#x521B;&#x5EFA;&#x5355;&#x901A;&#x9053;&#x63A9;&#x7801;&#x56FE;&#x50CF;
    //&#x6700;&#x5C0F;&#x5185;&#x63A5;&#x5706;&#x7B97;&#x6CD5;
    Point circleCenter(mask.cols / 2, mask.rows / 2);//&#x83B7;&#x53D6;&#x56FE;&#x50CF;&#x4E2D;&#x5FC3;&#x70B9;(&#x5706;&#x5FC3;)
    int radius = min(mask.cols, mask.rows) / 2;//&#x83B7;&#x53D6;&#x534A;&#x5F84;
    // &#x753B;&#x5706;
    circle(mask, circleCenter, radius, Scalar(255), -1);//&#x6784;&#x5EFA;&#x63A9;&#x7801;&#x56FE;&#x50CF;
    src.copyTo(dst, mask);//&#x63A9;&#x7801;&#x8FD0;&#x7B97;
    imshow("&#x63A9;&#x7801;", mask);
    imshow("&#x539F;&#x56FE;&#x50CF;", src);
    imshow("ROI&#x533A;&#x57DF;", dst);
    waitKey(0);
    return 0;
}

opencv——感兴趣区域(ROI)的分析和选取[详细总结]

2)第二种情形当我们不知道感兴趣ROI区域坐标时,我们通过鼠标交互地提取ROI。OpenCV中鼠标操作依赖鼠标的回调函数和响应函数实现。主函数中调用鼠标的回调函数,将鼠标操作与程序的窗口绑定,产生鼠标操作时回调函数调用鼠标响应函数执行。

🙄回调函数setMouseCallback

void setMouseCallback(const string& winname,
                      MouseCallback onMouse,
                      void* userdata=0 )

第一个参数,windows视窗名称,对名为winname的视窗进行鼠标监控;
第二个参数,鼠标响应处理函数,监听鼠标的点击,移动,松开,判断鼠标的操作类型,并进行响应的函数处理;
第三个参数,鼠标响应处理函数的ID,与鼠标相应处理函数相匹配就行,暂时只用到默认为0的情况。

😮 鼠标响应处理函数onMouse

void onMouse(int event,int x,int y,int flags,void *ustc)
&#x7B2C;&#x4E00;&#x4E2A;&#x53C2;&#x6570;&#xFF0C;&#x9F20;&#x6807;&#x64CD;&#x4F5C;&#x65F6;&#x95F4;&#x7684;&#x6574;&#x6570;&#x4EE3;&#x53F7;&#xFF0C;&#x5728;opencv&#x4E2D;&#xFF0C;event&#x9F20;&#x6807;&#x4E8B;&#x4EF6;&#x603B;&#x5171;&#x6709;10&#x4E2D;&#xFF0C;&#x4ECE;0-9&#x4F9D;&#x6B21;&#x4EE3;&#x8868;&#x5982;&#x4E0B;:

1EVENT_MOUSEMOVE      =0,    //&#x6ED1;&#x52A8;
2EVENT_LBUTTONDOWN    =1,    //&#x5DE6;&#x952E;&#x70B9;&#x51FB;
3EVENT_RBUTTONDOWN    =2,    //&#x53F3;&#x952E;&#x70B9;&#x51FB;
4EVENT_MBUTTONDOWN    =3,    //&#x4E2D;&#x95F4;&#x70B9;&#x51FB;
5EVENT_LBUTTONUP      =4,    //&#x5DE6;&#x952E;&#x91CA;&#x653E;
6EVENT_RBUTTONUP      =5,    //&#x53F3;&#x952E;&#x91CA;&#x653E;
7EVENT_MBUTTONUP      =6,    //&#x4E2D;&#x95F4;&#x91CA;&#x653E;
8EVENT_LBUTTONDBLCLK  =7,    //&#x5DE6;&#x952E;&#x53CC;&#x51FB;
9EVENT_RBUTTONDBLCLK  =8,    //&#x53F3;&#x952E;&#x53CC;&#x51FB;
10EVENT_MBUTTONDBLCLK  =9    //&#x4E2D;&#x95F4;&#x91CA;&#x653E;
&#x7B2C;&#x4E8C;&#x4E2A;&#x53C2;&#x6570;&#xFF0C;&#x4EE3;&#x8868;&#x9F20;&#x6807;&#x4F4D;&#x4E8E;&#x7A97;&#x53E3;&#x7684;(x&#xFF0C;y)&#x5750;&#x6807;&#x4F4D;&#x7F6E;&#xFF0C;&#x7A97;&#x53E3;&#x5DE6;&#x4E0A;&#x89D2;&#x9ED8;&#x8BA4;&#x4E3A;&#x539F;&#x70B9;&#xFF0C;&#x5411;&#x53F3;&#x4E3A;x&#x8F74;&#xFF0C;&#x5411;&#x4E0B;&#x4E3A;y&#x8F74;&#xFF1B;
&#x7B2C;&#x4E09;&#x4E2A;&#x53C2;&#x6570;&#xFF0C;&#x4EE3;&#x8868;&#x9F20;&#x6807;&#x7684;&#x62D6;&#x62FD;&#x4E8B;&#x4EF6;&#xFF0C;&#x4EE5;&#x53CA;&#x952E;&#x76D8;&#x9F20;&#x6807;&#x8054;&#x5408;&#x4E8B;&#x4EF6;&#xFF0C;&#x603B;&#x5171;&#x6709;32&#x79CD;&#x4E8B;&#x4EF6;&#xFF0C;&#x8FD9;&#x91CC;&#x4E0D;&#x518D;&#x8D58;&#x8FF0;&#x3002;
&#x7B2C;&#x56DB;&#x4E2A;&#x53C2;&#x6570;&#xFF0C;&#x51FD;&#x6570;&#x53C2;&#x6570;&#x7684;&#x7F16;&#x53F7;&#x3002;

🥰用onMouse实现手动截取ROI区域,自动提取ROI。代码如下:

using namespace std;
using namespace cv;
bool draw;
Mat src;//&#x539F;&#x59CB;&#x56FE;&#x50CF;
Mat roi;//ROI&#x56FE;&#x50CF;
Point cursor;//&#x521D;&#x59CB;&#x5750;&#x6807;
Rect rect;//&#x6807;&#x8BB0;ROI&#x7684;&#x77E9;&#x5F62;&#x6846;
void onMouse(int event, int x, int y, int flags, void *param);
int main(int argc, char** argv)
{

    src = imread("D:/opencv&#x7EC3;&#x4E60;&#x56FE;&#x7247;/&#x859B;&#x4E4B;&#x8C26;.jpg");
    namedWindow("SrcImage");
    imshow("SrcImage", src);
    setMouseCallback("SrcImage", onMouse, NULL);
    waitKey(0);
    return 0;
}
void onMouse(int event, int x, int y, int flags, void *param)
{
    Mat img = src.clone();
    switch (event)
    {
        //&#x6309;&#x4E0B;&#x9F20;&#x6807;&#x5DE6;&#x952E;
    case EVENT_LBUTTONDOWN:
        //&#x70B9;&#x51FB;&#x9F20;&#x6807;&#x56FE;&#x50CF;&#x65F6;&#xFF0C;&#x6E05;&#x9664;&#x4E4B;&#x524D;ROI&#x56FE;&#x50CF;&#x7684;&#x663E;&#x793A;&#x7A97;&#x53E3;
        destroyWindow("ROI");
        //&#x5B58;&#x653E;&#x8D77;&#x59CB;&#x5750;&#x6807;
        cursor = Point(x, y);
        //&#x521D;&#x59CB;&#x5316;&#x8D77;&#x59CB;&#x77E9;&#x5F62;&#x6846;
        rect = Rect(x, y, 0, 0);
        draw = true;
        break;

        //&#x677E;&#x5F00;&#x9F20;&#x6807;&#x5DE6;&#x952E;
    case EVENT_LBUTTONUP:
        if (rect.height > 0 && rect.width > 0)
        {
            //&#x5C06;img&#x4E2D;&#x7684;&#x77E9;&#x5F62;&#x533A;&#x57DF;&#x590D;&#x5236;&#x7ED9;roi&#xFF0C;&#x5E76;&#x663E;&#x793A;&#x5728;SignROI&#x7A97;&#x53E3;
            roi = img(Rect(rect.x, rect.y, rect.width, rect.height));
            rectangle(img, rect, Scalar(0, 0, 255), 2);
            namedWindow("SignROI");
            imshow("SignROI", img);

            //&#x5C06;&#x753B;&#x8FC7;&#x77E9;&#x5F62;&#x6846;&#x7684;&#x56FE;&#x50CF;&#x7528;&#x539F;&#x56FE;&#x50CF;&#x8FD8;&#x539F;
            src.copyTo(img);
            imshow("SrcImage", img);

            //&#x663E;&#x793A;ROI&#x56FE;&#x50CF;
            namedWindow("ROI");
            imshow("ROI", roi);
            waitKey(0);
        }
        draw = false;
        break;

        //&#x79FB;&#x52A8;&#x5149;&#x6807;
    case EVENT_MOUSEMOVE:
        if (draw)
        {
            //&#x7528;MIN&#x5F97;&#x5230;&#x5DE6;&#x4E0A;&#x70B9;&#x4F5C;&#x4E3A;&#x77E9;&#x5F62;&#x6846;&#x7684;&#x8D77;&#x59CB;&#x5750;&#x6807;&#xFF0C;&#x5982;&#x679C;&#x4E0D;&#x52A0;&#x8FD9;&#x4E2A;&#xFF0C;&#x753B;&#x77E9;&#x5F62;&#x65F6;&#x53EA;&#x80FD;&#x5411;&#x4E00;&#x4E2A;&#x65B9;&#x5411;&#x8FDB;&#x884C;
            rect.x = MIN(x, cursor.x);
            rect.y = MIN(y, cursor.y);
            rect.width = abs(cursor.x - x);
            rect.height = abs(cursor.y - y);
            //&#x9632;&#x6B62;&#x77E9;&#x5F62;&#x533A;&#x57DF;&#x8D85;&#x51FA;&#x56FE;&#x50CF;&#x7684;&#x8303;&#x56F4;
            rect &= Rect(0, 0, src.cols, src.rows);
        }
        break;
    }
}

参考链接:(8条消息) OpenCV中感兴趣区域的选取与检测(一)_鼹鼠的胡须 的博客-CSDN博客_opencv感兴趣区域

(8条消息) OpenCV探索之路(十三):详解掩膜mask_weixin_34221773的博客-CSDN博客

Original: https://blog.csdn.net/qq_44386034/article/details/125637882
Author: 明月清风_@
Title: opencv——感兴趣区域(ROI)的分析和选取[详细总结]

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

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

(0)

大家都在看

亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球