好久没有发OpenCV的博客了,最近想到了一个识别地图轮廓的方案,就写来试试。(识别中国的28个省份地图轮廓,不考虑直辖市)

首先,我的基本思路是  用最小的矩形将地图的轮廓圈出来,可以根据长方形的长宽比判断,也可将其缩放至特定的大小,计算其轮廓上的像素个数来判断。

缺点:用摄像头读取图片时,使用这种方法会有一些误差。

也可以ANN训练识别,但是这样做效率低。

step 1. 读取图片、处理图像

Mat src = imread("12.jpg");
Mat grayImage;
cvtColor(src, grayImage, CV_BGR2GRAY);
threshold(grayImage, grayImage, , , CV_THRESH_BINARY);
imshow("grayImage", grayImage);

问题来了,处理图片后的grayImage根本无法显示,结果为一张灰色的图片。

最后发现,因为大意,程序的最后没有加 cvWaitKey(0); 这句话,因此图片无法显示。

step 2. 寻找轮廓并画出

#include <opencv2/opencv.hpp>
#include <iostream>
using namespace cv;
using namespace std; int main()
{
Mat src = imread("timg.jpg");
Mat grayImage, dstImage;
src.copyTo(dstImage); int g_nStructElementSize = ; //结构元素(内核矩阵)的尺寸
//获取自定义核
Mat element = getStructuringElement(MORPH_RECT,
Size( * g_nStructElementSize + , * g_nStructElementSize + ),
Point(g_nStructElementSize, g_nStructElementSize));
erode(src, src, element);
cvtColor(src, grayImage, CV_BGR2GRAY);
threshold(grayImage, grayImage, , , CV_THRESH_BINARY);
imshow("2dst", grayImage); vector<vector<Point>> contours;
vector<Vec4i> hierarchy;
findContours(grayImage, contours, hierarchy, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_NONE);
int i = ;
Point2f pp[][];
vector<vector<Point>>::iterator It;
Rect rect[];
for (It = contours.begin(); It < contours.end(); It++){ //画出包围轮廓的最小矩形
Point2f vertex[];
rect[i] = boundingRect(*It);
vertex[] = rect[i].tl(); //矩阵左上角的点
vertex[].x = (float)rect[i].tl().x, vertex[].y = (float)rect[i].br().y; //矩阵左下方的点
vertex[] = rect[i].br(); //矩阵右下角的点
vertex[].x = (float)rect[i].br().x, vertex[].y = (float)rect[i].tl().y; //矩阵右上方的点 for (int j = ; j < ; j++)
line(dstImage, vertex[j], vertex[(j + ) % ], Scalar(, , ), );
}
imshow("dst", dstImage);
cvWaitKey();
return ;
}

结果发现根本找不到轮廓,最后发现原来是threshold函数参数设置错误,参数应如下:

threshold(grayImage, grayImage,,, THRESH_BINARY_INV);

同时由于地图边框线太细,应当先腐蚀图像,再二值化:

    int g_nStructElementSize = ; //结构元素(内核矩阵)的尺寸
///获取自定义核
Mat element = getStructuringElement(MORPH_RECT,
Size( * g_nStructElementSize + , * g_nStructElementSize + ),
Point(g_nStructElementSize, g_nStructElementSize));
erode(src, src, element);

step 3. 收集地图的数据后,用10个省得数据来检测

最后在不断的探索中,发现有两个数据可以作为一个地图的特征,即轮廓长宽比和轮廓面积与图片的像素数之比。

最后的代码如下:

#include <opencv2/opencv.hpp>
#include <iostream>
#include <math.h>
using namespace cv;
using namespace std; Mat result;
const double cha = 0.02; //可接受范围的误差 bool compare(double a, double b)
{
if (abs(a - b) < cha){
return true;
}
return false;
} bool result_output(double rate1,double rate2)
{
if (compare(rate1, (double) / ) && compare(rate2, 0.171524)){
cout << "陕西省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.270173)){
cout << "安徽省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.230148)){
cout << "福建省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.132584)){
cout << "甘肃省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.200146)){
cout << "广东省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.22718)){
cout <<"广西壮族自治区" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.219451)){
cout << "贵州省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.196616)){
cout << "海南省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.247134)){
cout << "河北省" << endl;
return true;
}
if (compare(rate1, (double) / ) && compare(rate2, 0.176323)){
cout << "河南省" << endl;
return true;
}
cout << "无法检测" << endl;
return false;
}
int main()
{
Mat src = imread("1.jpg");
Mat grayImage, dstImage;
src.copyTo(dstImage); int g_nStructElementSize = ; //结构元素(内核矩阵)的尺寸
///获取自定义核
Mat element = getStructuringElement(MORPH_RECT,
Size( * g_nStructElementSize + , * g_nStructElementSize + ),
Point(g_nStructElementSize, g_nStructElementSize));
erode(src, src, element);
cvtColor(src, grayImage, CV_BGR2GRAY);
blur(grayImage, grayImage, Size(, ));
threshold(grayImage, grayImage,,, THRESH_BINARY_INV);
grayImage.copyTo(result); vector< vector<Point> > contours;
vector<Vec4i> hierarchy;
findContours(grayImage, contours, hierarchy, CV_RETR_EXTERNAL, CV_CHAIN_APPROX_NONE);
drawContours(dstImage, contours, -, (, , ));
vector<Point> point = contours[]; Rect rect = boundingRect(point);
Point2f vertex[];
vertex[] = rect.tl();
vertex[].x = (float)rect.tl().x, vertex[].y = (float)rect.br().y;
vertex[] = rect.br();
vertex[].x = (float)rect.br().x, vertex[].y = (float)rect.tl().y;
for (int j = ; j < ; j++)
line(dstImage, vertex[j], vertex[(j + ) % ], Scalar(, , ), );
int x = rect.x, y = rect.y;
int h = rect.height, w = rect.width;
double rate = (double)h / w;
cout << "height:" << h << endl;
cout << "width:" << w << endl;
cout << "h / w:" << rate << endl;
double area = contourArea(point, false);
double sum = grayImage.cols * grayImage.rows;
cout << "面积:" << area << endl;
cout << "面积比:" << area / sum << endl; imshow("show", dstImage);
result_output(rate, area / sum); cvWaitKey();
return ;
}

最后发现一个问题,由于需要通过摄像头检测地图,图片可能会有一定角度的倾斜,因此应将Rect换成RotatedRect。

    RotatedRect  rect = minAreaRect(point);
Point2f vertex[];
rect.points(vertex);
for (int j = ; j < ; j++)
line(dstImage, vertex[j], vertex[(j + ) % ], Scalar(, , ), ); int h = rect.size.height, w = rect.size.width;

OpenCV——识别各省份地图轮廓的更多相关文章

  1. 【可视化】Echarts3 在世界地图中绘制中国各省份的轮廓

    要在世界地图展现出来的情况下绘制中国省份的轮廓,根据现有的echarts-api是不可行的. 但好在echarts也提供了自定义地图的方式,使用echarts.registerMap();来实现 第一 ...

  2. 转载:使用 OpenCV 识别 QRCode

    原文链接:http://coolshell.cn/articles/10590.html#jtss-tsina 识别二维码的项目数不胜数,每次都是开箱即用,方便得很. 这次想用 OpenCV 从零识别 ...

  3. 基于opencv 识别、定位二维码 (c++版)

    前言 因工作需要,需要定位图片中的二维码:我遂查阅了相关资料,也学习了opencv开源库.通过一番努力,终于很好的实现了二维码定位.本文将讲解如何使用opencv定位二维码. 定位二维码不仅仅是为了识 ...

  4. 基于Opencv识别,矫正二维码(C++)

    参考链接 [ 基于opencv 识别.定位二维码 (c++版) ](https://www.cnblogs.com/yuanchenhui/p/opencv_qr.html) OpenCV4.0.0二 ...

  5. opencv的实用研究--分析轮廓并寻找边界点

    opencv的实用研究--分析轮廓并寻找边界点 ​      轮廓是图像处理中非常常见的.对现实中的图像进行采样.色彩变化.灰度变化之后,能够处理得到的是“轮廓”.它直接地反应你了需要分析对象的边界特 ...

  6. OpenCV识别技术

    OpenCV识别技术# 老师:james 20181019 # 识别技术# Pycharm + Python3 + OpenCV """ 一.识别技术: 什么是OpenC ...

  7. jQuery中国各个省份地图分部代码

    jQuery中国各个省份地图分部代码 在线演示本地下载

  8. OpenCV——识别印刷体数字

    数字识别和其他的所有计算机视觉相关的应用都会分为两个步骤:ROI抽取和识别. 1. ROI抽取即将感兴趣的区域从原始图像中分离初来,这个步骤包括二值化,噪点的消除等2. 识别即通过一些分类器将第一步中 ...

  9. python opencv识别蓝牌车牌号 之 取出车牌号 (1/3)

    概述 车牌识别是计算机视频图像识别技术在车辆牌照识别中的一种应用,通常来讲如果结合opencv进行车牌识别主要分为四个大步骤,分别为: 图像采集 车牌定位 分割车牌字符 字符识别 当然,如果结合了机器 ...

随机推荐

  1. 定制库到Maven本地资源库

    这里有2个案例,需要手动发出Maven命令包括一个 jar 到 Maven 的本地资源库. 要使用的 jar 不存在于 Maven 的中心储存库中. 您创建了一个自定义的 jar ,而另一个 Mave ...

  2. c#基础操作

    内网 IPAddress ipAddr = Dns.Resolve(Dns.GetHostName()).AddressList[];//获得当前IP地址 string ip = ipAddr.ToS ...

  3. Js参数值中含有单引号或双引号解决办法

    <script type="text/javascript"> function Display(LoginEmail, UserName, ID) {         ...

  4. 安卓开发笔记——GridView组件

    1.什么是GridView? GridView(网格视图)是按照行列的方式来显示内容的,一般用于显示图片,图片等内容,比如实现九宫格图,用GridView是首选,也是最简单的. 2.正文 GridVi ...

  5. mongodb:修改oplog.rs 的大小size

    其内容字段说明: ts:操作日志的timestamp t: 未知? h:操作唯一随机值 v:oplog.rs的版本 op:操作类型: i:insert操作 u:update操作 d:delete操作 ...

  6. BarTender数据中的转义符序列知识讲解

    Datamatrix是二维码的一个成员,广泛用于商品的防伪.统筹标识.如果为 Data Matrix 条形码指定的“符号类型”不是 ECC 200,则将会启用“字符集”选项.Data Matrix 也 ...

  7. MathType如何设置标尺的单位

    MathType在编辑公式的时候,经常会需要将公式对齐.在将公式对齐的这个过程中,有时候会用到标尺,这样可以更精确的定位公式的位置.我们在使用标尺的时候,有时候会发现标尺上显示的是英寸,而我们平常已经 ...

  8. vs2012更改默认开发环境

    1.在菜单栏里找到“Tools”(工具),选择下面的“Import and Export Settings”(导入和导出设置),如图1所示:          图1 2.弹出如下界面,按提示选择你需要 ...

  9. okHttp3自用封装

    okHttp都已经出到3.2.0了,现在才开始要用到它,感觉自己好low~~ 根据平时自己的习惯,还是自己做一下封装,让代码撸起来更加顺畅一点! okhttp-3.2.0和okio-1.7.0就不多说 ...

  10. 用Eclipse编写Android程序的代码提示功能

    用Eclipse编写Android程序的代码提示功能主要是在java和xml文件中,有时候会失效,默认的提示功能有限. 1)java文件自动提示     Window->Preferences- ...