opencv检测直线方法之投影法

 更新时间:2018年12月26日 10:32:17   作者:恬梦  
这篇文章主要为大家详细介绍了opencv检测直线之投影法的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了opencv检测直线之投影法的具体代码,供大家参考,具体内容如下

以下是我对投影法的一点认识和实验:

投影法就是数字图像在某个方向上进行像素累加。通过水平和垂直方向的投影,可以得到表格图像投影的几个特点:

(1)表格区域的水平与竖直投影分布通常出现周期性的尖峰

(2)在文字投影的行与行之间或列与列之间常会出现明显的空白区

因此,求图像水平以及竖直投影,根据特点分别设以阈值就可以将横线以及竖直线所在位置确定。

  第一步:求图像的水平投影、竖直投影

  第二步:设定合理阈值,求取大于阈值的坐标(水平投影记录纵坐标,垂直投影记录横坐标)

  第三步:根据记录纵坐标恢复水平线,根据记录横坐标恢复竖直线。

下面附整体代码以及实验结果:

#include<iostream> 
#include<vector>
#include <cv.h> 
#include <highgui.h>
using namespace std;
using namespace cv;
Mat VerticalLine(Mat srcImageBin)//垂直线条检测 
{
 
 vector <int> array;//动态数组用来存储投影值大于阈值的横坐标
 
 int *colswidth = new int[srcImageBin.cols]; //申请src.image.cols个int型的内存空间,存储二值图中每列的白色像素数 
 memset(colswidth, 0, srcImageBin.cols * 4); //数组必须赋初值为零,否则出错。无法遍历数组。 
 
 int value;
 for (int i = 0; i < srcImageBin.cols; i++)
 {
 
 for (int j = 0; j < srcImageBin.rows; j++)
 {
 value = srcImageBin.at<uchar>(j, i);
 
 if (value == 255)
 {
 colswidth[i]++; //统计每列的白色像素点 
 
 }
 
 }
 
 }
 
 Mat lineImage(srcImageBin.rows, srcImageBin.cols, CV_8UC1, cv::Scalar(0, 0, 0));
 
 //寻找投影大于阈值0.3*srcImageBin.rows的横坐标
 for (int i = 0; i < srcImageBin.cols; i++)
 {
 bool flag = true;
 
 for (int j = 0; j < colswidth[i] && colswidth[i] >= (0.3*srcImageBin.rows); j++)
 {
 
 if (flag == true)
 {
 array.push_back(i);
 flag = false;
 }
 }
 }
 int count = array.size();
 //恢复直线
 for (int n = 0; n < srcImageBin.rows; n++)
 {
 for (int w = 0; w<count; w++)
 {
 if (srcImageBin.at<uchar>(n, array[w]) == 255)
 {
 lineImage.at<uchar>(n, array[w]) = 255;
 }
 
 }
 }
 
 
 
 delete[] colswidth;
 return lineImage;
}
Mat HorizonLine(Mat srcImageBin)//水平线条检测
{
 vector <int> array1;
 
 int *rowswidth = new int[srcImageBin.rows]; 
 memset(rowswidth, 0, srcImageBin.rows * 4); 
 int value;
 for (int i = 0; i < srcImageBin.rows; i++)
 {
 for (int j = 0; j < srcImageBin.cols; j++)
 {
 value = srcImageBin.at<uchar>(i, j);
 if (value == 255)
 {
 rowswidth[i]++; //统计每行的白色像素点 
 }
 }
 
 }
 
 Mat lineImage(srcImageBin.rows, srcImageBin.cols, CV_8UC1, cv::Scalar(0, 0, 0));
 
 //寻找投影大于阈值0.525*srcImageBin.cols的纵坐标
 for (int i = 0; i < srcImageBin.rows; i++)
 {
 bool flag = true;
 for (int j = 0; j < rowswidth[i] && rowswidth[i] >= (0.525*srcImageBin.cols); j++)
 {
 if (flag == true)
 {
 array1.push_back(i);
 flag = false;
 }
 
 }
 }
 int count = array1.size();
 
 //恢复水平线
 for (int h = 0; h<count; h++)
 {
 for (int m = 0; m < srcImageBin.cols; m++)
 {
 if (srcImageBin.at<uchar>(array1[h], m) == 255)
 {
 lineImage.at<uchar>(array1[h], m) = 255;
 }
 
 }
 }
 
 
 delete[] rowswidth;//释放前面申请的空间 
 return lineImage;
}
int main()
{
 Mat srcImage = imread("E:\\x.jpg");
 Mat closeimage;
 imshow("原图", srcImage);
 if (srcImage.channels() > 1)
 cvtColor(srcImage, srcImage, CV_RGB2GRAY);
 Mat srcImageBin;
 threshold(srcImage, srcImageBin, 140, 255, CV_THRESH_OTSU | CV_THRESH_BINARY_INV);
 
 Mat VP;
 VP = VerticalLine(srcImageBin);
 
 Mat HP;
 HP = HorizonLine(srcImageBin);
 
 Mat mergelineImage;
 
 bitwise_or(HP, VP, mergelineImage);
 imshow("mergelineImage", mergelineImage);
 
 waitKey(0);
 return 0;
 
}

实验结果如下:

由上结果可知,如果直线中间有字会被误检为直线,图中用红色椭圆标出。

文中若有错误的不妥的地方,还望指出,以便共同学习。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • C++11中的default函数使用

    C++11中的default函数使用

    这篇文章主要介绍了C++11中的default函数使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-05-05
  • Qt图形图像开发之曲线图表库QChart编译安装详细方法与使用实例

    Qt图形图像开发之曲线图表库QChart编译安装详细方法与使用实例

    这篇文章主要介绍了Qt图形图像开发之曲线图表库QChart编译安装详细方法与使用实例,需要的朋友可以参考下
    2020-03-03
  • C++中strstr函数的实现方法总结

    C++中strstr函数的实现方法总结

    这篇文章主要介绍了C++中strstr函数的实现方法总结的相关资料,希望通过本文能帮助到大家,让大家掌握这部分内容,需要的朋友可以参考下
    2017-10-10
  • 如何基于 Blueprint 在游戏中创建实时音视频功能

    如何基于 Blueprint 在游戏中创建实时音视频功能

    我们在本文先来讲讲如何在 Unreal 中用 Blueprint 快速实现。稍后会分享基于 C++的实现步骤。感兴趣的朋友跟随小编一起看看吧
    2020-05-05
  • wxWidgets自定义按钮的方法

    wxWidgets自定义按钮的方法

    这篇文章主要为大家详细介绍了wxWidgets自定义按钮的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-02-02
  • C语言利用EasyX绘制小企鹅表情包

    C语言利用EasyX绘制小企鹅表情包

    这篇文章主要为大家详细介绍了C语言如何利用EasyX绘图库实现绘制可爱的小企鹅表情包,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-12-12
  • C语言中大小端问题实例探索解决方法

    C语言中大小端问题实例探索解决方法

    这篇文章主要介绍了C语言中大小端问题实例,总的来说这并不是一道难题,那为什么要拿出这道题介绍?拿出这道题真正想要传达的是解题的思路,以及不断优化探寻最优解的过程。希望通过这道题能给你带来一种解题优化的思路
    2023-02-02
  • C++类型转换的深入总结

    C++类型转换的深入总结

    这篇文章主要给大家介绍了关于C++类型转换的深入总结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-12-12
  • 解析C++编程中异常相关的堆栈展开和throw()异常规范

    解析C++编程中异常相关的堆栈展开和throw()异常规范

    这篇文章主要介绍了C++编程中异常相关的堆栈展开和throw()异常规范,throw()规范部分文中结合了C++11标准的新特性来讲,需要的朋友可以参考下
    2016-01-01
  • 基于Matlab制作一个数独求解器

    基于Matlab制作一个数独求解器

    这篇文章主要为大家详细介绍了如何利用Matlab制作一个数独求解器,文中的示例代码讲解详细,对我们学习Matlab有一定帮助,需要的可以参考一下
    2022-05-05

最新评论