opencv3/C++轮廓的提取与筛选方式
轮廓提取
findContours发现轮廓
findContours( InputOutputArray binImg, //输入8bit图像,0值像素值不变,非0的像素看成1;(变为二值图像) OutputArrayOfArrays contours,//输出找到的轮廓对象 OutputArray, hierachy// 图像的拓扑结构 int mode, //轮廓返回的模式(RETR_TREE等) int method,//发现方法(CHAIN_APPROX_SIMPLE等) Point offset=Point()//轮廓像素的位移(默认没有位移(0, 0)) )
【报错问题】
findContours()有时会报告“已触发了一个断点”等错误,尝试过y有效的解决方法有:
1.为vector提前申请一定的空间,如
std::vector<std::vector<Point>> contours(500)
2.Debug版切换为Release版;
drawContours绘制轮廓
drawContours( InputOutputArray binImg, // 输出图像 OutputArrayOfArrays contours,//找到的全部轮廓对象 Int contourIdx//轮廓索引号 const Scalar & color,//绘制颜色 int thickness,//绘制线宽 int lineType ,//线的类型(默认8) InputArray hierarchy,//拓扑结构图 int maxlevel,//最大层数(0只绘制当前的,1表示绘制绘制当前及其内嵌的轮廓) Point offset=Point()//轮廓位移 )
示例:
#include<opencv2/opencv.hpp> using namespace cv; int main() { Mat src,dst; src = imread("E:/image/image/shape.jpg"); if(src.empty()) { printf("can not load image \n"); return -1; } namedWindow("input", CV_WINDOW_AUTOSIZE); imshow("input", src); dst = Mat::zeros(src.size(), CV_8UC3); blur(src,src,Size(3,3)); cvtColor(src,src,COLOR_BGR2GRAY); Canny(src, src, 20, 80, 3, false); std::vector<std::vector<Point>> contours; std::vector<Vec4i> hierarchy; findContours(src, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0)); RNG rng(0); for(int i = 0; i < contours.size(); i++) { Scalar color = Scalar(rng.uniform(0,255), rng.uniform(0,255), rng.uniform(0,255)); drawContours(dst, contours, i, color, 2, 8, hierarchy, 0, Point(0,0)); } namedWindow("output", CV_WINDOW_AUTOSIZE); imshow("output",dst); waitKey(); return 0; }
使用opencv3时(测试用opencv3.1.0)发现,cv命名空间下没有了vector,而在opencv2中(测试用opencv2.4.10)还存在。后查看各自的头文件发现:
opencv.hpp头文件中包含着core.hpp(#include “opencv2/core.hpp”);
而在opencv2的core.hpp中包含有
........ #include <map> #include <new> #include <string> #include <vector> .......
等头文件,但在opencv3的core.hpp中删去这些包含项。
因此在使用opencv3时cv命名空间下没有了vector。
使用opencv2.4.10时可以写:
#include<opencv2/opencv.hpp> using namespace cv; int main() { Mat src,dst; src = imread("E:/image/image/shape.jpg"); if(src.empty()) { printf("can not load image \n"); return -1; } namedWindow("input", CV_WINDOW_AUTOSIZE); imshow("input", src); dst = Mat::zeros(src.size(), CV_8UC3); blur(src,src,Size(5,5)); Canny(src, src, 20, 80, 3, false); vector<vector<Point>>contours; vector<Vec4i> hierarchy; findContours(src, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0)); RNG rng(0); for(int i = 0; i < contours.size(); i++) { Scalar color = Scalar(rng.uniform(0,255), rng.uniform(0,255), rng.uniform(0,255)); drawContours(dst, contours, i, color, 2, 8, hierarchy, 0, Point(0,0)); } namedWindow("output", CV_WINDOW_AUTOSIZE); imshow("output",dst); waitKey(); return 0; }
绘制轮廓外矩形框
常用绘制轮廓外形状的函数:
cv::boundingRect(InputArray points)绘制一个矩形(轮廓周围最小矩形左上角点和右下角点)
cv::minAreaRect(InputArray points)绘制轮廓周围最小旋转矩形
cv::minEnclosingCircle(InputArray points, Point2f& center, float& radius)//绘制轮廓周围最小圆形
cv::fitEllipse(InputArray points)绘制轮廓周围最小椭圆
绘制轮廓外矩形框:
#include<opencv2/opencv.hpp> using namespace cv; int main() { Mat src,dst; src = imread("E:/image/shape.jpg"); if(src.empty()) { printf("can not load image \n"); return -1; } namedWindow("input", CV_WINDOW_AUTOSIZE); imshow("input", src); dst = Mat::zeros(src.size(), CV_8UC3); std::vector<std::vector<Point>>contours; std::vector<Vec4i> hierarchy; blur(src,src,Size(3,3)); cvtColor(src,src,COLOR_BGR2GRAY); Canny(src, src, 20, 80, 3, false); findContours(src, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0)); RNG rng(0); std::vector<std::vector<Point>> contoursPloy(contours.size()); std::vector<RotatedRect> minRects(contours.size()); for(int i = 0; i < contours.size(); i++) { minRects[i] = minAreaRect(Mat(contours[i])); Scalar color = Scalar(rng.uniform(0,255), rng.uniform(0,255), rng.uniform(0,255)); drawContours(dst, contoursPloy, i, color, 1,8,std::vector<Vec4i>(), 0, Point(0, 0)); Point2f rectPoints[4]; minRects[i].points(rectPoints); for (int j = 0; j < 4; j++) { line(dst, rectPoints[j], rectPoints[(j+1)%4], color, 1, 8, 0); } } namedWindow("output", CV_WINDOW_AUTOSIZE); imshow("output",dst); waitKey(); return 0; }
轮廓筛选
moments( InputArray array,//输入数据 bool binaryImage=false //是否为二值图像 ) contourArea( InputArray contour,//输入轮廓数据 bool oriented//返回绝对值(默认false) ) arcLength( InputArray curve,//输入轮廓 bool closed// 轮廓否是封闭曲线 )
轮廓筛选示例:
使用轮廓的面积和长度特征对轮廓进行筛选后用外接矩形将筛选后的轮廓框选出来。
#include<opencv2/opencv.hpp> using namespace cv; void trackBar(int,void*); Mat src,dst; std::vector<std::vector<Point>>contours; int area = 0, length = 0; int main() { src = imread("E:/image/shape.jpg"); if(src.empty()) { printf("can not load image \n"); return -1; } namedWindow("input", CV_WINDOW_AUTOSIZE); imshow("input", src); dst = Mat::zeros(src.size(), CV_8UC3); std::vector<Vec4i> hierarchy; blur(src,dst,Size(3,3)); cvtColor(dst,dst,COLOR_BGR2GRAY); Canny(dst, dst, 20, 80, 3, false); findContours(dst, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0)); namedWindow("output", CV_WINDOW_AUTOSIZE); createTrackbar("area:", "output", &area,150,trackBar); createTrackbar("length:", "output", &length,150,trackBar); waitKey(); return 0; } void trackBar(int,void*) { Mat src1 = src.clone(); RNG rng(0); std::vector<std::vector<Point>> contoursPloy(contours.size()); std::vector<RotatedRect> minRects(contours.size()); for(int i = 0; i < contours.size(); i++) { if(contourArea(contours[i]) > area && arcLength(contours[i], false) > length) { minRects[i] = minAreaRect(Mat(contours[i])); Scalar color = Scalar(rng.uniform(0,255), rng.uniform(0,255), rng.uniform(0,255)); //drawContours(dst, contoursPloy, i, color, 1,8,vector<Vec4i>(), 0, Point(0, 0)); Point2f rectPoints[4]; minRects[i].points(rectPoints); for (int j = 0; j < 4; j++) { line(src1, rectPoints[j], rectPoints[(j+1)%4], color, 2, 8, 0); } } } imshow("output",src1); src1 = src; }
以上这篇opencv3/C++轮廓的提取与筛选方式就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。
栏 目:C语言
本文地址:https://www.xiuzhanwang.com/a1/Cyuyan/101.html
您可能感兴趣的文章
- 04-02c语言没有round函数 round c语言
- 01-10深入理解C++中常见的关键字含义
- 01-10使用C++实现全排列算法的方法详解
- 01-10c++中inline的用法分析
- 01-10用C++实现DBSCAN聚类算法
- 01-10全排列算法的非递归实现与递归实现的方法(C++)
- 01-10C++大数模板(推荐)
- 01-10浅谈C/C++中的static与extern关键字的使用详解
- 01-10深入C/C++浮点数在内存中的存储方式详解
- 01-10深入理解C/C++混合编程
阅读排行
本栏相关
- 04-02c语言函数调用后清空内存 c语言调用
- 04-02func函数+在C语言 func函数在c语言中
- 04-02c语言的正则匹配函数 c语言正则表达
- 04-02c语言用函数写分段 用c语言表示分段
- 04-02c语言中对数函数的表达式 c语言中对
- 04-02c语言编写函数冒泡排序 c语言冒泡排
- 04-02c语言没有round函数 round c语言
- 04-02c语言分段函数怎么求 用c语言求分段
- 04-02C语言中怎么打出三角函数 c语言中怎
- 04-02c语言调用函数求fibo C语言调用函数求
随机阅读
- 08-05dedecms(织梦)副栏目数量限制代码修改
- 04-02jquery与jsp,用jquery
- 01-10C#中split用法实例总结
- 01-10使用C语言求解扑克牌的顺子及n个骰子
- 08-05DEDE织梦data目录下的sessions文件夹有什
- 08-05织梦dedecms什么时候用栏目交叉功能?
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打开原生自带读写NTFS功能(图文
- 01-11ajax实现页面的局部加载
- 01-10SublimeText编译C开发环境设置