opencv检测直线方法之投影法
作者:恬梦 发布时间:2023-08-28 04:43:15
标签: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;
}
实验结果如下:
由上结果可知,如果直线中间有字会被误检为直线,图中用红色椭圆标出。
文中若有错误的不妥的地方,还望指出,以便共同学习。
来源:https://blog.csdn.net/u013972657/article/details/77744302


猜你喜欢
- 芬兰数学家因卡拉花费3个月设计出了世界上迄今难度最大的数独游戏,而且它只有一个答案。因卡拉说只有思考能力最快、头脑最聪明的人才能破解这个游戏
- 下面给大家介绍几种比较常见的解决办法,具体内容如下:1.有时候eclipse不自动编译,把project clean一下,让R.java重新
- 方法重载概述方法重载指同一个类中定义的多个方法之间的关系,满足下列条件的多个方法互相构成重载* 多个方法在同一个类中* 多个放方法具有相同方
- 前言我们都知道在java中进行日期格式化使用simpledateformat。通过格式 yyyy-MM-dd 等来进行格式化,但是你知道其中
- 本文实例讲述了C#判断一天、一年已经过了百分之多少的方法。分享给大家供大家参考。具体如下:这里写了四个函数,分别是1.判断当前时间过了今天的
- 前天在做批量数据导入新增时,要对数据进行有效性判断,其中还要去除重复,如果没出现linq的话可能会新声明一个临时对象集合,然后遍历原始数据判
- 前言上一篇我们介绍了 Animation 和 AnimationController 的使用,这是最
- 一、流程和任务的关系以下是一个简单的请假流程图,其中有一个开始事件,两个用户任务,一个结束事件。启动流程后,activiti会自动创建第一个
- 前言一直用C#开发程序,.NET的功能越来越多,变化也挺大的,从最初的封闭,到现在的开源,功能不断的增加,一直在进步。下面就来给大家详细介绍
- 相信大家对SaaS架构都有所了解,这里也不过多介绍,让我们直奔主题。技术框架springboot版本为2.3.4.RELEASE持久层采用J
- 在文章中,我们将对输入到机器学习模型中的数据集进行预处理。这里我们将对一个硬币数据集进行预处理,以便以后在监督学习模型中进行训练。在机器学习
- 承蒙各位厚爱,我们一起每天进步一点点!(鼠标选中空白处查看答案)1、以下不属于构造方法特征的是()正确答案: D构造方法名与类名相同构造方法
- 这两天看到Hibernate的代理部分,第一反应是底层使用了反射,针对用户实体生成了代理类,后来反应过来了,反射没有任何可以产生新类的能力,
- 已知两个链表list1和list,2,各自非降序排列,将它们合并成另外一个链表list3,并且依然有序,要求保留所有节点。实现过程中,lis
- 今天写项目突然出现了无法启动Gradle的bug,如下图然后就看了log日志:这个问题是我第一次看见,然后就开始了各种百度,有说需要在And
- 一、认识AdapterViewFilpper AdapterViewFilpper 继承 了Adapte
- 秒杀功能秒杀场景现在已经非常常见了,各种电商平台都有秒杀的产品,接下来我们模拟一个秒杀的项目,最终能够确保高并发下的线程安全。界面比较简单,
- 国家气象局提供了三种数据的形式网址在:http://www.weather.com.cn/data/sk/101010100.htmlhtt
- 最近在用SpringMvc做Http接口时,对方在调用我接口时发现Date格式的默认转化为long,因此在前端页面看到的是一串数字。我们可以
- 1.如果执行了try块没有异常,则继续运行finally块中的语句,即使try块通过return,break,或者continue于最后的语