opencv 图像轮廓
图片解析:
原图:
code:
#include <opencv\cv.h>
#include <opencv\highgui.h>
#include <opencv\cxcore.h>
#include <stdlib.h>
#include <stdio.h> int main(int argc, char* argv[]){ #if 1 int i, j; CvMemStorage* storage = cvCreateMemStorage(0);
//以单通道方式加载图像
IplImage* img = cvLoadImage("C:\\Users\\zxl\\Desktop\\0.png",CV_LOAD_IMAGE_GRAYSCALE);
//常见一个等大小的3通道图像 (作为备份)
IplImage* imgColor = cvCreateImage(cvGetSize(img),8,3);
//用来显示轮廓的
IplImage* contoursImage = cvCreateImage(cvSize(img->width,img->height),8,1); CvSeq* contours = 0 , *contoursTemp=0;
cvZero(contoursImage); //对图像进行二值化
cvThreshold(img,img,100,255,CV_THRESH_BINARY); //img的备份
cvCvtColor(img,imgColor,CV_GRAY2BGR); // 提取图像img的轮廓信息函数 contours指向第一个轮廓 将轮廓存放在CvMemStorage类型的变量storage里面
//&contours =>表示指针 指向能够提取的第一个轮廓
//CV_RETR_CCOMP =>表示轮廓的排列方式 有4种
//-------------------------------------
//CV_RETR_EXTERNAL 只返回最外面的轮廓
//first=c0
//CV_RETR_CCOMP 把外轮廓用双向链表的方式存放,有顺序 <从里到外><从右到左>
//frist=c01001 <-> c01000 <-> c010 <-> c000 <-> c0
// | | |
// h0100 h0000 h01 <-> h00
//CV_RETR_LIST 所有的轮廓(无分内外轮廓)通过1个链表的方式存储,<从里到外><从右到左>
//first=c01001 <-> c01000 <-> h0100 <-> h0000 <-> c010 <-> c000 <-> h01 <-> h00 <-> c0
//
//
//CV_RETR_TREE 按照树形结构存储,
//first = c0
// |
// h00 <-> h01
// | |
// c000 c010
// | |
// h0000 c01000 <-> c01001
//-------------------------------------
//CV_CHAIN_APPROX_NONE=> 轮廓显示是坐标的形式还是点的形式
//提取轮廓后源图像会发生变化 所以需要用到开始生命的备份机制 /*
//备份机制检查原图
cvNamedWindow( "1");
cvShowImage( "1", img );
*/ int total = cvFindContours( img, storage, &contours, sizeof(CvContour),
CV_RETR_CCOMP, CV_CHAIN_APPROX_NONE, cvPoint(0,0) ); /*
//备份机制检查提取后
cvNamedWindow( "2");
cvShowImage( "2", img );
cvWaitKey(0);
*/ //提取指针所指的地方
contoursTemp = contours ; int count=0; //对轮廓进行循环
//h_next 表示水平的方向上轮廓链接
//v_next 表示垂直方向上的轮廓链接
for(;contoursTemp!=0 ; contoursTemp=contoursTemp->h_next){ //通过这个循环访问每一个轮廓
//提取外轮廓 上的所以坐标点
for( i=0; i<contoursTemp->total; i++) { //通过这个循环 得到坐标
CvPoint * pt = (CvPoint*)cvGetSeqElem(contoursTemp, i); // 读出第i个点。
//轮廓窗口让其显示为白色
cvSetReal2D(contoursImage , pt->y , pt->x , 255.0);
//在原窗口让其显示为红色
cvSet2D(imgColor,pt->y,pt->x,cvScalar(0,0,255,0));
}
count++; //提取内轮廓上的所以坐标点
CvSeq* InterCon = contoursTemp->v_next; for(;InterCon!=0;InterCon=InterCon->h_next){
for(i=0;i<InterCon->total;i++) {
CvPoint * pt = (CvPoint*)cvGetSeqElem(InterCon, i); // 读出第i个点。 cvSetReal2D(contoursImage , pt->y , pt->x , 255.0);
cvSet2D(imgColor,pt->y,pt->x,cvScalar(0,255,0,0));
}
}
} cvNamedWindow( "image", 1 );
cvShowImage( "image", imgColor ); cvNamedWindow( "contours");
cvShowImage("contours",contoursImage); cvWaitKey(0); cvReleaseMemStorage( &storage );
cvReleaseImage( &img );
cvReleaseImage(&contoursImage);
cvReleaseImage(&imgColor); #endif return 0;
}
效果:
opencv 图像轮廓的更多相关文章
- OpenCV图像轮廓检测
轮廓检测: 轮廓检测的原理通俗的说就是掏空内部点,比如原图中有3*3的矩形点.那么就可以将中间的那一点去掉. 一.关键函数1.1 cvFindContours函数功能:对图像进行轮廓检测,这个函数将 ...
- opencv 6 图像轮廓与图像分割修复 3 图像的矩,分水岭,图像修补
图像的矩 矩的计算:moments()函数 计算轮廓面积:contourArea()函数 #include "opencv2/highgui/highgui.hpp" #inclu ...
- OpenCV计算机视觉学习(8)——图像轮廓处理(轮廓绘制,轮廓检索,轮廓填充,轮廓近似)
如果需要处理的原图及代码,请移步小编的GitHub地址 传送门:请点击我 如果点击有误:https://github.com/LeBron-Jian/ComputerVisionPractice 1, ...
- OpenCV笔记(3)(Canny边缘检测、高斯金字塔、拉普拉斯金字塔、图像轮廓、模板匹配)
一.Canny边缘检测 Canny边缘检测是一系列方法综合的结果.其中主要包含以下步骤: 1.使用高斯滤波器,平滑图像,滤除噪声. 2.计算图像中每个像素点的梯度强度和方向. 3.应用非极大值抑制(N ...
- 【python+opencv】轮廓发现
python+opencv---轮廓发现 轮廓发现---是基于图像边缘提取的基础寻找对象轮廓的方法, 所有边缘提取的阈值选定会影响最终轮廓发现的结果. 介绍两种API使用: -cv.findConto ...
- OpenCV—Python 轮廓检测 绘出矩形框(findContours\ boundingRect\rectangle
千万注意opencv的轮廓检测和边缘检测是两码事 本文链接:https://blog.csdn.net/wsp_1138886114/article/details/82945328 1 获取轮廓 O ...
- OpenCV3入门(十)图像轮廓
1.图像轮廓 1.1图像轮廓与API函数 轮廓是一系列相连的点组成的曲线,代表了物体的基本外形,相对于边缘,轮廓是连续的,边缘并不全部连续.一般地,获取图像轮廓要经过下面几个步骤: 1) 读取 ...
- OpenCV图像处理中“投影技术”的使用
本文区分"问题引出"."概念抽象"."算法实现"三个部分由表及里具体讲解OpenCV图像处理中"投影技术" ...
- opencv——图像直方图与反向投影
引言 在图像处理中,对于直方图这个概念,肯定不会陌生.但是其原理真的可以信手拈来吗? 本文篇幅有点长,在此列个目录,大家可以跳着看: 分析图像直方图的概念,以及opencv函数calcHist()对于 ...
随机推荐
- Android如何在ListView中嵌套ListView
前几天因为项目的需要,要在一个ListView中放入另一个ListView,也即在一个ListView的每个ListItem中放入另外一个ListView.但刚开始的时候,会发现放入的小ListVie ...
- 各种不会。。。。编译和安装hadoop过程中好多命令和工具不会
http://blog.csdn.net/bamuta/article/details/13506893 64位下解决方法(重新编译~~) http://zhidao.baidu.com/link? ...
- 【HDOJ】4426 Palindromic Substring
综合性很强的一道题目,结合manacher,后缀数组,哈希,RMQ,二分可解.基本思路是通过manacher可以找到所有可能的回文串,哈希去重,后缀数组二分找数目.最后暴力求解.需要注意kth需要为_ ...
- Android开发之AIDL的使用一--跨应用启动Service
启动其他App的服务,跨进程启动服务. 与启动本应用的Service一样,使用startService(intent)方法 不同的是intent需要携带的内容不同,需要使用intent的setComp ...
- POI根据EXCEL模板,修改内容导出新EXCEL (只支持HSSF)
package excelPoiTest; import java.io.File; import java.io.FileInputStream; import java.io.FileOutput ...
- 原创js模型驱动
使用ajax方式请求数据,向页面展示一个对象的时候,往往让人头疼的是一大堆 .val() .text()方法,这样做固然不会出错,但是效率太低 以下提供一个自己编写的Jquery模型驱动插件,正在测 ...
- RPi 2B GPIO 测试
/************************************************************************************** * RPi 2B GPI ...
- 装饰器模式(Decorator)
转自http://blog.csdn.net/hust_is_lcd/article/details/7884320 1.认识装饰器模式 装饰模式能够实现动态的为对象添加功能,是从一个对象外部来给对象 ...
- android LayoutInflater和inflate()方法的用法(转载)
原文出处:http://www.cnblogs.com/top5/archive/2012/05/04/2482328.html 在实际开发中LayoutInflater这个类还是非常有用的,它的作用 ...
- RHCS集群
理论基础: User → HA → Lb → web → sql → 分布式filesystem ->磁盘I/O 用户 高可用 负载均衡 应用 数据库 mf ...