作者:翟天保Steven
版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处

需求说明

在对图像进行处理时,经常会有这类需求:客户想要提取出图像中某条直线、圆线或者ROI区域内的感兴趣数据,进行重点关注。该需求在图像检测领域尤其常见。ROI区域一般搭配Rect即可完成提取,直线和圆线数据的提取没有现成的函数,需要自行实现。

直线的提取见:

OpenCV-获取图像中直线上的数据_翟天保的博客-CSDN博客

而圆线的提取则是本文要将的内容,对圆线而言,将线上某点作为起点,沿顺时针或逆时针方向依次提取感兴趣数据,可放置在容器中。那么如何快速提取呢?本文提供了一种比较简单的思路,应用窗口模板,在窗口中快速找到下一可前进点的位置,步进然后再找下个点,形成路径追踪,进而实现整圈圆线数据的提取。

具体流程

1)初始化。设置路径追踪窗口尺寸size为3,创建path作为行进路径,p点作为起点,c用来存放目标数据点。

cv::Mat c;
int size = 3;
cv::Mat path = mask.clone();
Point p = Point(center.x + radius, center.y);

2)将起点放置在c中,将path中的起点值置0,表示该点已经走过。

c.push_back(src.at<uchar>(p.y, p.x));
path.at<uchar>(p.y, p.x) = 0;

3)用WinDataNum函数判断当前点的窗口内有几个可前进点,若无则说明路径封死或者完成路径,wn值表示可前进点的个数。

int wn = WinDataNum(path, p, size);

4)窗口内遍历,查看是否有可前进路径,若找到,则将当前点信息刷新为此点,并将标记符find设为true,find的意义是快速中断遍历,用来提速。

int t = size / 2;
bool find = false;
for (int i = p.y - t; i <= p.y + t; ++i)
{uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255){p.x = j;p.y = i;find = true;break;}}if (find)break;
}

5)若找到了点,即find为true,则将该点的数据存放在c中,path中置0,并以该点为中心搜索窗口内可前进路径。

if (find)
{c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;wn = WinDataNum(path, p, size);
}
elsebreak;

6)若wn为0了,则说明路径封死或者完成路径了,跳出循环,函数执行完毕。

while (wn)
{int t = size / 2;bool find = false;for (int i = p.y - t; i <= p.y + t; ++i){uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255){p.x = j;p.y = i;find = true;break;}}if (find)break;}if (find){c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;wn = WinDataNum(path, p, size);}elsebreak;}

功能函数

// 获取圆圈上的数据,逆时针存储,起点在中心同行最右侧数据
cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius)
{cv::Mat c;int size = 3;cv::Mat path = mask.clone();Point p = Point(center.x + radius, center.y);c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;int wn = WinDataNum(path, p, size);while (wn){int t = size / 2;bool find = false;for (int i = p.y - t; i <= p.y + t; ++i){uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255){p.x = j;p.y = i;find = true;break;}}if (find)break;}if (find){c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;wn = WinDataNum(path, p, size);}elsebreak;}return c;
}
// 获取窗口内的有效数据个数
int WinDataNum(cv::Mat path, cv::Point p, int size)
{int number = 0;int t = size / 2;for (int i = p.y - t; i <= p.y + t; ++i){uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255)number++;}}return number;
}

C++测试代码

#include <iostream>
#include <opencv2/opencv.hpp>
#include <string>using namespace std;
using namespace cv;cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius);
int WinDataNum(cv::Mat path, cv::Point p, int size);int main()
{cv::Mat src = imread("test.jpg", 0);cv::Mat mask = cv::Mat::zeros(src.size(), CV_8UC1);cv::Point center = cv::Point(src.cols / 2, src.rows / 2);int radius = min(src.cols, src.rows) / 2 - 10;circle(mask, center, radius, Scalar(255), 1, 8);cv::Mat c = getCircleData(src, mask, center, radius);src.setTo(0, mask == 0);imshow("src", src);imshow("mask", mask);waitKey(0);return 0;
}// 获取圆圈上的数据,逆时针存储,起点在中心同行最右侧数据
cv::Mat getCircleData(cv::Mat src, cv::Mat mask, cv::Point center, int radius)
{cv::Mat c;int size = 3;cv::Mat path = mask.clone();Point p = Point(center.x + radius, center.y);c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;int wn = WinDataNum(path, p, size);while (wn){int t = size / 2;bool find = false;for (int i = p.y - t; i <= p.y + t; ++i){uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255){p.x = j;p.y = i;find = true;break;}}if (find)break;}if (find){c.push_back(src.at<uchar>(p.y, p.x));path.at<uchar>(p.y, p.x) = 0;wn = WinDataNum(path, p, size);}elsebreak;}return c;
}// 获取窗口内的有效数据个数
int WinDataNum(cv::Mat path, cv::Point p, int size)
{int number = 0;int t = size / 2;for (int i = p.y - t; i <= p.y + t; ++i){uchar *g = path.ptr<uchar>(i);for (int j = p.x - t; j <= p.x + t; ++j){if (g[j] == 255)number++;}}return number;
}

测试效果

图1 原图

图2 掩膜内图像

如图1图2所示,掩膜内的图像数据就是我们要提取的目标。

图3 放大后数据搜索路径

图3放大后可以看出,起点是230,之后的数据是230、231、236、232、234、146等等,再看c容器中的数据。

图4 容器内数据

对比完开头,再看结尾,如图3所示,是234、234、231、234、234,然后就是起点230,查看容器。

图5 容器内数据

这样有的小伙伴可能觉得中间会不会有数据错误呢,很简单,打开VS复制代码后,搭配ImageWatch插件,debug调试打断点观察path矩阵,看看它的255数据是不是按预想的路径消失,如果是则说明扔的数据也没有问题。

本文提供的只是一个简单思路,有一定局限性。比如该方法在圆线宽为1时效果最佳,若线宽大了就不能用窗口简单判断了;另外,起点在右侧时是逆时针获取数据,起点在左侧时是顺时针获取数据,如果想统一标准的话,最好加上起点的位置判断,然后决定是否将c的数据翻转。至于运行速度方面,3000*3000的图像矩阵中运行基本为0ms,毕竟只是提取了一条圆线而已。

如果函数有什么可以改进完善的地方,非常欢迎大家指出,一同进步何乐而不为呢~

如果文章帮助到你了,可以点个赞让我知道,我会很快乐~加油!

OpenCV-获取图像中圆线上的数据相关推荐

  1. OpenCV-获取图像中直线上的数据

    作者:翟天保Steven 版权声明:著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处 需求说明 在对图像进行处理时,经常会有这类需求:客户想要提取出图像中某条直线或者ROI区域内的感 ...

  2. OpenCv获取图像中椭圆长短轴的点的位置

        目的是得到长短轴上的灰度变化曲线 1 首先使用Opencv的  FindContour   fitEllipse  可以得到旋转矩形 2 然后获取矩形长轴和短轴的位置,然后根据位置得到灰度值, ...

  3. 草稿 爬虫-访问登陆可见的页面-请求时带上cookie数据

    小结 在发起请求时,在请求头中带上cookie数据 然后访问登陆后可见的网页 那么就可以实现 目标,获取博客后台草稿文章数量 文章管理-CSDN博客 https://mp.csdn.net/ 未登陆的 ...

  4. 如何在Tableau中的县级上映射数据

    地理空间数据可视化(Geospatial Data Visualization) 总览(Overview) The geography of counties and county-level equ ...

  5. webform空间在html输出数据库,如何:在 ASP 上的数据网格中导出数据。 Microsoft Excel 的 NET WebForm...

    启动 Visual Studio .NET. 在"文件"菜单上,指向"新建",然后单击"项目". 在 "项目类型" 窗格 ...

  6. OpenCV将GIS数据加载到OpenCV容器中的实例(附完整代码)

    OpenCV将GIS数据加载到OpenCV容器中的实例 OpenCV将GIS数据加载到OpenCV容器中的实例 OpenCV将GIS数据加载到OpenCV容器中的实例 #include "o ...

  7. 操作XML 报错:根级别上的数据无效 和 给定编码中的字符无效 解决办法

    根级别上的数据无效,解决如下: private void button1_Click(object sender, EventArgs e) { try { XmlDocument doc = new ...

  8. 使用Python批量筛选上千个Excel文件中的某一行数据并另存为新Excel文件(下篇)

    点击上方"Python爬虫与数据挖掘",进行关注 回复"书籍"即可获赠Python从入门到进阶共10本电子书 今 日 鸡 汤 野火烧不尽,春风吹又生. 大家好, ...

  9. linux下读取ntfs数据,在Linux中读取NTFS分区上的数据

    在Linux中读取NTFS分区上的数据 在有些情况下,系统需要访问本地NTFS分区上的数据,也可能需要访问网络上NTFS文件格式的数据.而使用mount -t挂载文件系统时,系统报告不支持NTFS错误 ...

最新文章

  1. Go 知识点(15)— 切片长度和容量
  2. 你还在为 TCP 重传、滑动窗口、流量控制、拥塞控制发愁吗?看完图解就不愁了...
  3. pandas使用to_dict方法把datafraem保存为字典形式
  4. Windows Server 2012 克隆修改SID
  5. ICCV11 Distributed Cosegmentation via Submodular Optimization on Anisotropic Diffusion
  6. 电容屏和电阻屏的区别
  7. 由程序猿yyyy-MM-dd跨年Bug引发的深思
  8. php生成五星红旗,php基于GD库画五星红旗的方法_PHP
  9. 【音视频—基础】分辨率、码率和帧率
  10. 服务器ie浏览器总是未响应怎么办,IE8浏览器总是无响应或卡死崩溃怎么办
  11. Java程序员如何通过阿里、百度的招聘面试
  12. EChart饼图文字大小调整
  13. K8S查看容器日志、进入容器内部
  14. 【字节跳动2019年】算法岗笔试题
  15. 【音乐可视化】Audacity,一款免费的多轨音频编辑器
  16. xlsx无法导入MySQL?
  17. 红米k30支持html,红米k30支持多大的快充
  18. //3.求和:n+nn+nnn+nnnn+nnnnn
  19. Idea取消光标所在行颜色
  20. 【小程序源码合集】云开发前后端完整代码,大学生校园社团体管理活动预约微信小程序源码

热门文章

  1. 【资料下载】Python 第九讲——灵活运用docker,实现深度学习的环境搭建...
  2. Confluence 6 数据收集隐私策略
  3. MyBatsi-Mapper映射文件
  4. 如何从0开发一个Atom组件
  5. JDBC连接错误:通过端口 1433 连接到主机 localhost 的 TCP/IP 连接失败。。。
  6. [PhoneGap]Mac下搭建PhoneGap开发环境
  7. php截取字符串,无乱码
  8. 黄聪:UEditor如何在wordpress中调用
  9. 使用surface初体验
  10. Cisco PVLAN的配置