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

简介: OpenCV-获取图像中直线上的数据

需求说明

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


      当直线为纵向或者横向时,比较简单,只需要从起点到终点提取该行或者列的数据即可;但是直线若为斜向的,则需要从起点出发,向终点方向逐个像素提取。大家都知道,图像是由许多像素组成,而斜向直线的数据提取路线并不一定就是标准的斜线,也可能是呈阶梯状的路线,而如何进行路线设计,就是本文所要展示的内容。

具体流程

      1)建立vector<pair<float,int>> result容器用于存放数据,设置初始化参数。其中,inImage是输入图像,start为起点,end为终点,点的类型为cv::Point。

vector<pair<float, int>> result;
int row = inImage.rows;
int col = inImage.cols;
int r1 = start.y;
int c1 = start.x;
int r2 = end.y;
int c2 = end.x;

      2)确定两点间距离dist,将起点到终点的横坐标差和纵坐标差进行勾股定理可得。所得距离可能为带小数的数据,然而像素的个数都为整数,所以进行四舍五入。除此之外,还要判断下距离,若距离为0,则只返回起点数据。

float dist = round(sqrt(pow(float(r2) - float(r1), 2.0) + pow(float(c2) - float(c1), 2.0)));
if (dist <= 0.00001f) {
  pair<float, int> temp;
  temp.first = inImage.at<float>(r1, c1);
  temp.second = 0;
  result.push_back(temp);
  return result;
}

  3)确定横向纵向的步进间隔。

float slope_r = (float(r2) - float(r1)) / dist;
float slope_c = (float(c2) - float(c1)) / dist;

      4)建立Flag地图,用于标记已存储过的位置,避免同一数据二次放入。

cv::Mat Flag = cv::Mat::zeros(mask.size(), mask.type());

     5)开始存储数据。计数从0开始,若该点处于掩膜内,且Flag地图中没有标记,则进行存储。

int k = 0;
for (float i = 0; i <= dist; ++i) {
  // 若该点处于掩膜内,且未被Flag存储,则进行存储工作
  if ((mask.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 255)
    && (Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 0))
  {
    pair<float, int> temp;
    temp.first = inImage.at<float>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c)));
    temp.second = k;
    Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) = 255;
    k++;
    result.push_back(temp);
  }
}

功能函数

/**
 * @brief GetOneDimLineData                 获取一维直线数据
 * @param inImage                           输入位相图
 * @param mask                              输入掩膜图
 * @param start                             起始点坐标
 * @param end                               终点坐标
 * @return                                  直线数据(数值&序号)
 */
vector<pair<float, int>> GetOneDimLineData(const cv::Mat inImage, cv::Mat mask, cv::Point start, cv::Point end)
{
  vector<pair<float, int>> result;
  int row = inImage.rows;
  int col = inImage.cols;
  int r1 = start.y;
  int c1 = start.x;
  int r2 = end.y;
  int c2 = end.x;
  // 确定两点间距离
  float dist = round(sqrt(pow(float(r2) - float(r1), 2.0) + pow(float(c2) - float(c1), 2.0)));
  if (dist <= 0.00001f) {
    pair<float, int> temp;
    temp.first = inImage.at<float>(r1, c1);
    temp.second = 0;
    result.push_back(temp);
    return result;
  }
  // 横向纵向的步进间隔
  float slope_r = (float(r2) - float(r1)) / dist;
  float slope_c = (float(c2) - float(c1)) / dist;
  // Flag地图,用于存储已放入的数据,避免同一数据二次放入
  cv::Mat Flag = cv::Mat::zeros(mask.size(), mask.type());
  // 数据量计数,从0开始
  int k = 0;
  for (float i = 0; i <= dist; ++i) {
    // 若该点处于掩膜内,且未被Flag存储,则进行存储工作
    if ((mask.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 255)
      && (Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 0))
    {
      pair<float, int> temp;
      temp.first = inImage.at<float>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c)));
      temp.second = k;
      Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) = 255;
      k++;
      result.push_back(temp);
    }
  }
  return result;
}

C++测试代码

#include <iostream>
#include <opencv2/opencv.hpp>
#include <opencv2/highgui.hpp>
using namespace std;
using namespace cv;
vector<pair<float, int>> GetOneDimLineData(const cv::Mat inImage, cv::Mat mask, cv::Point start, cv::Point end);
int main()
{
  Mat src(10,10,CV_32FC1,nan(""));
  for (int i = 3; i < 7; ++i)
  {
    for (int j = 3; j < 9; ++j)
    {
      src.at<float>(i, j) = rand() % 255;
    }
  }
  cv::Mat mask = cv::Mat::zeros(src.size(), CV_8UC1);
  mask.setTo(255, src == src);
  Point start = Point(2, 1);
  Point end = Point(8, 7);
  vector<pair<float, int>> test= GetOneDimLineData(src,mask, start, end);
  cout << "size:" << test.size() << endl;
  for (int i=0;i<test.size();++i)
  {
    cout << i << ":" << endl;
    cout << test[i].first << " " << test[i].second << endl;
  }
  return 0;
}
/**
 * @brief GetOneDimLineData                 获取一维直线数据
 * @param inImage                           输入位相图
 * @param mask                              输入掩膜图
 * @param start                             起始点坐标
 * @param end                               终点坐标
 * @return                                  直线数据(数值&序号)
 */
vector<pair<float, int>> GetOneDimLineData(const cv::Mat inImage, cv::Mat mask, cv::Point start, cv::Point end)
{
  vector<pair<float, int>> result;
  int row = inImage.rows;
  int col = inImage.cols;
  int r1 = start.y;
  int c1 = start.x;
  int r2 = end.y;
  int c2 = end.x;
  // 确定两点间距离
  float dist = round(sqrt(pow(float(r2) - float(r1), 2.0) + pow(float(c2) - float(c1), 2.0)));
  if (dist <= 0.00001f) {
    pair<float, int> temp;
    temp.first = inImage.at<float>(r1, c1);
    temp.second = 0;
    result.push_back(temp);
    return result;
  }
  // 横向纵向的步进间隔
  float slope_r = (float(r2) - float(r1)) / dist;
  float slope_c = (float(c2) - float(c1)) / dist;
  // Flag地图,用于存储已放入的数据,避免同一数据二次放入
  cv::Mat Flag = cv::Mat::zeros(mask.size(), mask.type());
  // 数据量计数,从0开始
  int k = 0;
  for (float i = 0; i <= dist; ++i) {
    // 若该点处于掩膜内,且未被Flag存储,则进行存储工作
    if ((mask.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 255)
      && (Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) == 0))
    {
      pair<float, int> temp;
      temp.first = inImage.at<float>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c)));
      temp.second = k;
      Flag.at<uchar>(int(r1) + int(round(i * slope_r)), int(c1) + int(round(i * slope_c))) = 255;
      k++;
      result.push_back(temp);
    }
  }
  return result;
}

测试效果

图1 初始化测试图像

图2 Flag地图

图3 结果打印

      不难看出,获取的数据为直线上数据。对于有一定斜度的直线,Flag地图可能呈现阶梯状步进路线,这也是正常的~


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


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

相关文章
|
1月前
|
计算机视觉
Opencv学习笔记(三):图像二值化函数cv2.threshold函数详解
这篇文章详细介绍了OpenCV库中的图像二值化函数`cv2.threshold`,包括二值化的概念、常见的阈值类型、函数的参数说明以及通过代码实例展示了如何应用该函数进行图像二值化处理,并展示了运行结果。
322 0
Opencv学习笔记(三):图像二值化函数cv2.threshold函数详解
|
2月前
|
算法 计算机视觉
opencv图像形态学
图像形态学是一种基于数学形态学的图像处理技术,它主要用于分析和修改图像的形状和结构。
49 4
|
3月前
|
计算机视觉 Windows Python
windows下使用python + opencv读取含有中文路径的图片 和 把图片数据保存到含有中文的路径下
在Windows系统中,直接使用`cv2.imread()`和`cv2.imwrite()`处理含中文路径的图像文件时会遇到问题。读取时会返回空数据,保存时则无法正确保存至目标目录。为解决这些问题,可以使用`cv2.imdecode()`结合`np.fromfile()`来读取图像,并使用`cv2.imencode()`结合`tofile()`方法来保存图像至含中文的路径。这种方法有效避免了路径编码问题,确保图像处理流程顺畅进行。
317 1
|
2月前
|
存储 计算机视觉
Opencv的基本操作(一)图像的读取显示存储及几何图形的绘制
本文介绍了使用OpenCV进行图像读取、显示和存储的基本操作,以及如何绘制直线、圆形、矩形和文本等几何图形的方法。
Opencv的基本操作(一)图像的读取显示存储及几何图形的绘制
|
3月前
|
算法 计算机视觉 Python
python利用opencv进行相机标定获取参数,并根据畸变参数修正图像附有全部代码(流畅无痛版)
该文章详细介绍了使用Python和OpenCV进行相机标定以获取畸变参数,并提供了修正图像畸变的全部代码,包括生成棋盘图、拍摄标定图像、标定过程和畸变矫正等步骤。
python利用opencv进行相机标定获取参数,并根据畸变参数修正图像附有全部代码(流畅无痛版)
WK
|
3月前
|
编解码 计算机视觉 Python
如何在OpenCV中进行图像转换
在OpenCV中,图像转换涉及颜色空间变换、大小调整及类型转换等操作。常用函数如`cvtColor`可实现BGR到RGB、灰度图或HSV的转换;`resize`则用于调整图像分辨率。此外,通过`astype`或`convertScaleAbs`可改变图像数据类型。对于复杂的几何变换,如仿射或透视变换,则可利用`warpAffine`和`warpPerspective`函数实现。这些技术为图像处理提供了强大的工具。
WK
108 1
|
4月前
|
机器学习/深度学习 XML 计算机视觉
OpenCV(Open Source Computer Vision Library)是一个开源的计算机视觉和机器学习库,它提供了大量的函数和工具,用于处理图像和视频数据。
OpenCV(Open Source Computer Vision Library)是一个开源的计算机视觉和机器学习库,它提供了大量的函数和工具,用于处理图像和视频数据。
|
5月前
|
算法 计算机视觉
【Qt&OpenCV 图像的感兴趣区域ROI】
【Qt&OpenCV 图像的感兴趣区域ROI】
165 1
|
5月前
|
运维 算法 计算机视觉
【Qt&OpenCV 图像的模板匹配 matchTemplate/minMaxLoc】
【Qt&OpenCV 图像的模板匹配 matchTemplate/minMaxLoc】
77 1
|
5月前
|
存储 编解码 算法
【Qt&OpenCV 检测图像中的线/圆/轮廓 HoughLinesP/HoughCircles/findContours&drawContours】
【Qt&OpenCV 检测图像中的线/圆/轮廓 HoughLinesP/HoughCircles/findContours&drawContours】
89 0