实验七 查找并绘制轮廓实验

  • 一、实验目的和要求
  • 二、实验内容
  • 三、实验仪器、设备
  • 四、实验原理
  • 五、实验步骤
  • 六、实验注意事项
  • 七、实验结果
  • 八、实验总结


一、实验目的和要求

  理解查找图像轮廓的基本原理;掌握使用OpenCV实现查找轮廓的代码编写方法;掌握使用OpenCV实现绘制轮廓的代码编写方法。

二、实验内容

  (一)新建工程;
  (二)在Vs2015中配置OpenCV;
  (三)使用OpenCV中的findContours函数实现查找轮廓;
  (四)使用OpenCV中的drawContours函数实现绘制轮廓。

三、实验仪器、设备

  计算机一台,已安装 Windows7操作系统和Visual Studio 2015

四、实验原理

  (一)虽然Canny之类的边缘检测算法可以根据像素之间的差异,检测出轮廓边界的像素,但是它并没有将轮廓作为一个整体。所以,本实验便是把这些边缘像素组装成轮廓。
  (二)一个轮廓一般对应一系列的点,也就是图像中的一条曲线。其表示方法可能根据不同的情况而有所不同。在OpenCV中,可以用findContours函数从二值图像中查找轮廓。drawContours函数用于在图像中绘制外部或内部轮廓 。

五、实验步骤

  (一)创建Visual Studio 2015控制台程序;
  (二)在Visual Studio 2015中配置OpenCV;
  (三)编写代码,使用findContours函数实现实现查找轮廓;
  (四)编写代码,使用drawContours函数实现实现绘制轮廓。

六、实验注意事项

  (一)完成OpenCV安装之后,VS中配置OpenCV的方法;
  (二)findContours函数的功能和使用方法;
  (三)drawContours函数的功能和使用方法。

七、实验结果

  (一)实验代码

//----------------------------【头文件、命名空间包含部分】----------------------------
//		描述:包含程序所使用的头文件和命名空间
//-------------------------------------------------------------------------------------
#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>
using namespace cv;
using namespace std;


//----------------------------【宏定义部分】-------------------------------------------- 
//		描述:定义一些辅助宏 
//------------------------------------------------------------------------------------- 
#define WINDOW_NAME1 "【原始图窗口】"			//为窗口标题定义的宏 
#define WINDOW_NAME2 "【轮廓图】"					//为窗口标题定义的宏 


//----------------------------【全局变量声明部分】--------------------------------------
//		描述:全局变量的声明
//-------------------------------------------------------------------------------------
Mat g_srcImage; 
Mat g_grayImage;
int g_nThresh = 80;
int g_nThresh_max = 255;
RNG g_rng(12345);
Mat g_cannyMat_output;
vector<vector<Point>> g_vContours;
vector<Vec4i> g_vHierarchy;


//----------------------------【全局函数声明部分】--------------------------------------
//		描述:全局函数的声明
//-------------------------------------------------------------------------------------
static void ShowHelpText( );
void on_ThreshChange(int, void* );


//---------------------------【main( )函数】--------------------------------------------
//		描述:控制台应用程序的入口函数,我们的程序从这里开始执行
//-------------------------------------------------------------------------------------
int main( int argc, char** argv )
{
	//【0】改变console字体颜色
	system("color 1F"); 

	//【0】显示欢迎和帮助文字
	ShowHelpText( );

	// 加载源图像
	g_srcImage = imread( "1.jpg", 1 );
	if(!g_srcImage.data ) { printf("读取图片错误,请确定目录下是否有imread函数指定的图片存在~! \n"); return false; } 

	// 转成灰度并模糊化降噪
	cvtColor( g_srcImage, g_grayImage, COLOR_BGR2GRAY );
	blur( g_grayImage, g_grayImage, Size(3,3) );

	// 创建窗口
	namedWindow( WINDOW_NAME1, WINDOW_AUTOSIZE );
	imshow( WINDOW_NAME1, g_srcImage );

	//创建滚动条并初始化
	createTrackbar( "canny阈值", WINDOW_NAME1, &g_nThresh, g_nThresh_max, on_ThreshChange );
	on_ThreshChange( 0, 0 );

	waitKey(0);
	return(0);
}

//-----------------------------【on_ThreshChange( )函数】------------------------------  
//      描述:回调函数
//-------------------------------------------------------------------------------------  
void on_ThreshChange(int, void* )
{

	// 用Canny算子检测边缘
	Canny( g_grayImage, g_cannyMat_output, g_nThresh, g_nThresh*2, 3 );
	imshow("canny", g_cannyMat_output);

	// 寻找轮廓
	//void findContours(InputOutputArray image, OutputArrayOfArrays contours,
	//	OutputArray hierarchy, int mode,
	//	int method, Point offset = Point());
	findContours( g_cannyMat_output, g_vContours, g_vHierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0) );

	// 绘出轮廓
	Mat drawing = Mat::zeros( g_cannyMat_output.size(), CV_8UC3 );
	for( int i = 0; i< g_vContours.size(); i++ )
	{
		Scalar color = Scalar( g_rng.uniform(0, 255), g_rng.uniform(0,255), g_rng.uniform(0,255) );//任意值
		drawContours( drawing, g_vContours, i, color, 2, 8, g_vHierarchy, 0, Point() );
	}

	// 显示效果图
	imshow( WINDOW_NAME2, drawing );
}


//----------------------------【ShowHelpText( )函数】----------------------------------  
//      描述:输出一些帮助信息  
//-------------------------------------------------------------------------------------  
static void ShowHelpText()  
{  
	//输出欢迎信息和OpenCV版本
	printf("\n\n\t\t\t非常感谢购买《OpenCV3编程入门》一书!\n");
	printf("\n\n\t\t\t此为本书OpenCV3版的第70个配套示例程序\n");
	printf("\n\n\t\t\t   当前使用的OpenCV版本为:" CV_VERSION );
	printf("\n\n  ----------------------------------------------------------------------------\n");
	//输出一些帮助信息  
	printf(   "\n\n\t欢迎来到【在图形中寻找轮廓】示例程序~\n\n");  
	printf(   "\n\n\t按键操作说明: \n\n"  
		"\t\t键盘按键任意键- 退出程序\n\n"  
		"\t\t滑动滚动条-改变阈值\n" );  
}

  (二)显示结果

openCV视觉寻线 opencv视觉检测_c++


openCV视觉寻线 opencv视觉检测_OpenCV_02


openCV视觉寻线 opencv视觉检测_openCV视觉寻线_03


openCV视觉寻线 opencv视觉检测_计算机视觉_04


openCV视觉寻线 opencv视觉检测_OpenCV_05

八、实验总结

  本次实验的主要内容是理解查找图像轮廓的基本原理;掌握使用OpenCV实现查找轮廓的代码编写方法;掌握使用OpenCV实现绘制轮廓的代码编写方法。新建工程,在VS2015中配置OpenCV,使用OpenCV中的findContours函数实现查找轮廓;使用OpenCV中的drawContours函数实现绘制轮廓。学会了用findContours函数从二值图像中查找轮廓。drawContours函数用于在图像中绘制外部或内部轮廓。