《Opencv3编程入门》学习笔记—第八章

news2025/1/9 15:05:14

《Opencv3编程入门》学习笔记

记录一下在学习《Opencv3编程入门》这本书时遇到的问题或重要的知识点。

第八章 图像轮廓与图像分割修复

一、查找并绘制轮廓

一个轮廓一般对应一系列的点,也就是图像中的一条曲线,其表示方法可能根据不同的情况而有所不同。在OpenCV中可以用findContours()函数从二值图像中查找轮廓。

(一)寻找轮廓:findContours()函数

findContours()函数用于在二值图像中寻找轮廓

void findContours(InputOutputArray image, OutputArrayOfArrays contours, OutputArray hierarchy, int mode, int method, Point offset=Point())

(1)第一个参数:InputArray类型的image,输入图像,必须是二值图像。我们可以用compare()、imrange()、threshold()、adaptivethreshold()、canny()等函数由灰度图或者彩色图创建二值图像。
(2)第二个参数:此参数类型可以使用vector<vector>类型,用于存放监测到的轮廓, 每一个轮廓被储存为一个点构成的容器。contour[i]表示第i个轮廓。
(3)第三个参数:可以使用vector类型。该容器的元素为4维int类型的向量。向量中的4个元素分别存放当前轮廓contour[i]的后一个轮廓编号、前一个轮廓编号、父轮廓编号、内嵌轮廓编号。
(4)第四个参数:int类型,轮廓检索模式,取值如下表所示

  • RETR_EXTERNAL 只检测外层轮廓
  • RETR_LIST 检测所有轮廓,轮廓间不建立等级关系
  • RETE_CCOMP 检测所有轮廓,并且将其组织为双层结构,顶层为连通域的外围边界,次层为孔的内层边界
  • RETE_TREE 提取所有轮廓,并建立网状的轮廓结构

(5)第五个参数:int类型,轮廓的近似方法,取值如下表

  • CHAIN_APPROX_NONE 获取每个轮廓的每个像素,相邻的两个像素位置差不超过1
  • CHAIN_APPROX_SIMPLE 压缩水平方向,垂直方向、对角线方向的元素,只保留该方向的终点坐标,例如一个矩形轮廓只需要4个点
  • CHAIN_APPROX_TC89_L1, CHAIN_APPROX_TC89_KCOS 使用teh-Chinl链逼近算法

(6)第六个参数:Point类型的偏移量,每个轮廓点的可选偏移量。对于ROI(感兴趣区域)图像中找出的轮廓,并要在整个图像中进行分析时,这个参数便可以派上用场。

(二)绘制轮廓:drawContours()函数

void drawContours(InputOutputArray image, InputArrayOfArrays contours, int contourIdx, const Scalar& color, int thickness=1, int lineType=8, InputArray hierarchy=noArray(), int maxLevel=INT_MAX, Point offset=Point() )

(1)第一个参数:image表示目标图像,Mat类型即可。
(2)第二个参数:contours表示输入的轮廓组,每一组轮廓由点vector构成,可以使用vector<vector>类型。
(3)第三个参数:contourIdx指明画第几个轮廓,如果该参数为负值,则画全部轮廓,
(4)第四个参数:color为轮廓的颜色,Scalar类型。
(5)第五个参数:thickness为轮廓的线宽,如果为负值或CV_FILLED表示填充轮廓内部。
(6)第六个参数:lineType为线型,取值类型如下表:
在这里插入图片描述
(7)第七个参数:可选的轮廓结构信息,
(8)第八个参数:为maxLevel,绘制轮廓的最大等级
(9)第九个参数:Point类型的偏移量。

(三)基础示例程序:轮廓查找

示例代码

#include <opencv2/opencv.hpp>
#include <opencv2/highgui/highgui.hpp>
#include <opencv2/imgproc/imgproc.hpp>
using namespace cv;
using namespace std;

int main()
{
	//以灰度值模式载入原始图像
	Mat srcImage = imread("D://lili/Desktop/jpg/opencv/8.jpg", 0);
	imshow("原始图", srcImage);

	//初始化结果图
	Mat dstImage = Mat::zeros(srcImage.rows, srcImage.cols, CV_8UC3);

	//src取大于阈值120的那部分
	srcImage = srcImage > 120;
	imshow("阈值处理后的原始图", srcImage);

	//定义轮廓和层次结构
	vector<vector<Point>> contours;
	vector<Vec4i> hierarchy;            //该容器内放了4维int向量

	//查找轮廓
	findContours(srcImage, contours, hierarchy, RETR_CCOMP, CHAIN_APPROX_SIMPLE);

	//遍历所有顶层的轮廓,以随机颜色绘制出每个连接组件的颜色
	int index = 0;
	for(; index >= 0; index = hierarchy[index][0])
	{
		Scalar color(rand() & 255, rand() & 255, rand() & 255);
		drawContours(dstImage, contours, index, color, 1, 8, hierarchy);
	}

	imshow("轮廓图", dstImage);
	waitKey(0);
	return 0;
}

运行效果
在这里插入图片描述
在这里插入图片描述

(四)综合示例程序:查找并绘制轮廓

示例代码

#include <opencv2/opencv.hpp>
#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, g_grayImage, g_cannyMat_output;
int g_nThresh=80, g_nThresh_max=255;
RNG g_rng(12345);
vector<vector<Point>> g_vContour;
vector<Vec4i> g_vHierarchy;

//全局函数声明
void on_threshChange(int, void*);


int main()
{
	g_srcImage = imread("D://lili/Desktop/jpg/opencv/11.jpg", 1);
	if(!g_srcImage.data)
	{
		cout << "读取图片错误" << endl;
		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;
}


//回调函数
void on_threshChange(int, void*)
{
	//用canny算子检测边缘
	Canny(g_grayImage, g_cannyMat_output, g_nThresh, g_nThresh * 2, 3);

	//寻找轮廓
	findContours(g_cannyMat_output, g_vContour, 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_vContour.size(); i++)
	{
		Scalar color = Scalar(g_rng.uniform(0, 255), g_rng.uniform(0, 255), g_rng.uniform(0, 255));
		drawContours(drawing, g_vContour, i, color, 2, 8, g_vHierarchy, 0, Point());
	}
	imshow(WINDOW_NAME2, drawing);
}

运行效果
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

二、寻找物体的凸包

(一)凸包

  • 简单来说凸包就是给定二维平面上的点集,将最外层点连接起来构成的凸多边形。
  • 理解物体形状或轮廓的一种比较有用的方法是计算一个物体的凸包,然后计算其凸缺陷(convexity defects)例如,图中A-H区域是凸包的各个”缺陷”:
    在这里插入图片描述

(二)寻找凸包:convexHull()函数

void convexHull(InputArray points,OutputArray hull,bool clockwise = false,bool returnPoints = true)

(1)参数一:输入的二维点集,Mat类型或std::vector
(2)参数二:输出参数,找到的凸包,返回的hull是points中点的索引
(3)参数三:操作方向标识符,为真时输出凸包为顺时针方向,否则逆时针
(4)参数四:操作标识符,默认true,标志为真时函数返回各凸包的各个点,否则返回凸包各点的指数,输出数组是std:vector时此标志忽略

(三)基础示例程序:凸包检测基础

示例代码
随机生成3-103个坐标值随机的彩色点,然后利用convexHull,对由这些点链接起来的图像求凸包。

#include<opencv2/opencv.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<opencv2/imgproc/imgproc.hpp>
using namespace cv;
using namespace std;
int main()
{
	//初始化变量和随机值
	Mat image(600, 600, CV_8UC3);
	RNG& rng = theRNG();
	//循环,按下ESC退出程序,否则有键按下便一直更新
	while (1)
	{
		//参数初始化
		char key;
		int count = (unsigned)rng % 100 + 3;//随机生成点的数量
		vector<Point>points;//点值
		//随机生成点坐标
		for (int i = 0; i < count; i++)
		{
			Point point;
			point.x = rng.uniform(image.cols / 4, image.cols * 3 / 4);
			point.y = rng.uniform(image.rows / 4, image.rows * 3 / 4);
			points.push_back(point);
		}
		//检测凸包
		vector<int>hull;
		convexHull(Mat(points), hull, true);//返回的hull是points中点的索引
		//绘制出随机颜色的点
		image = Scalar::all(0);
		for (int i = 0; i < count; i++)
		{
			circle(image, points[i], 3, Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 1, 1);
		}
		//准备参数
		int hullcount = (int)hull.size();//凸包边数
		Point point0 = points[hull[hullcount - 1]];//连接凸包边的坐标点
		//绘制凸包的边
		for (int i = 0; i < hullcount; i++)
		{
			Point point = points[hull[i]];
			line(image, point0, point, Scalar(255,255,255), 2, 1);
			point0 = point;
		}
		//显示效果图
		imshow("凸包检测示例", image);
		//按下ESC,程序退出
		key = (char)waitKey();
		if (key == 27) break;
	}
	return 0;
}

运行效果
在这里插入图片描述

(四)综合示例程序:寻找和绘制物体的凸包

示例代码

/*
效果:
	滑动条控制阈值g_nThresh,以改变g_thresholdImage_output,findContours以此参数为输入,查找不同轮廓图,从而得到不同凸包检测效果图
*/
#include<opencv2/opencv.hpp>
#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, g_grayImage, g_thresholdImage_output, g_dstImage;
int g_nThresh = 80;
int g_nThresh_max = 255;
RNG g_rng(12345);
Mat srcImage_copy = g_srcImage.clone();
vector<vector<Point>>g_vContours;
vector<Vec4i>g_vHierarchy;
//全局函数
static void ShowHelpText();
static void on_ThreshChange(int, void*);
int main()
{
	//改变console字体颜色
	system("color 1F");
	ShowHelpText();

	//载入原图
	g_srcImage = imread("D://lili/Desktop/jpg/opencv/11.jpg", 1);
	if (!g_srcImage.data)
	{
		printf("图片载入失败\n");
		return false;
	}
	//创建窗口
	namedWindow(WINDOW_NAME1, WINDOW_AUTOSIZE);
	namedWindow(WINDOW_NAME2, WINDOW_AUTOSIZE);
	imshow(WINDOW_NAME1, g_srcImage);

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

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

	waitKey(0);
	return 0;
}
static void on_ThreshChange(int, void*)
{
	//对图像进行二值化,控制阈值
	threshold(g_grayImage, g_thresholdImage_output, g_nThresh, 255, THRESH_BINARY);
	//查找轮廓
	findContours(g_thresholdImage_output, g_vContours, g_vHierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
	//遍历每个轮廓,寻找其凸包
	vector<vector<Point>>hull(g_vContours.size());
	for (int i = 0; i < g_vContours.size(); i++)
	{
		convexHull(Mat(g_vContours[i]), hull[i], false);
	}
	//绘制轮廓及其凸包
	Mat g_dstImage = Mat::zeros(g_thresholdImage_output.size(), CV_8UC3);
	//或 for (int index = 0; index >= 0; index = g_vHierarchy[index][0])
	for (int index = 0; index < g_vContours.size(); index++)
	{
		//或 Scalar color(g_rng.uniform(0, 255), g_rng.uniform(0, 255), g_rng.uniform(0, 255));//任意值
		Scalar color(rand() & 255, rand() & 255, rand() & 255);
		drawContours(g_dstImage, g_vContours, index, color, 1, 8, vector<Vec4i>(), 0, Point());
		drawContours(g_dstImage, hull, index, color, 1, 8, vector<Vec4i>(), 0, Point());
	}
	//显示效果图
	imshow(WINDOW_NAME2, g_dstImage);
}
static void ShowHelpText()
{
	printf("\n\n\t欢迎来到【在图形中寻找轮廓及其凸包】示例程序!\n\n");
	printf("\n\n\t操作说明:\n\n");
	printf("\t\t键盘任意键-退出程序\n\n");
	printf("\t\t滑动滚动条-改变阈值\n");
}

运行效果
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

三、使用多边形将轮廓包围

(一)返回外部矩形边界:boundingRect()函数

(1)作用:返回指定点集最外面的边界矩形(四个顶点)
(2)函数原型:Rect boundingRect(InputArray points)

(二)寻找最小包围矩形:minAreaRect()函数

(1)作用:返回指定点集可旋转的最小面积的包围矩形(四个顶点)
(2)函数原型:RotatedRect minAreaRect(InputArray points)

(三)寻找最小包围圆形:minEnclosingCircle()函数

(1)作用:利用一种迭代算法,返回指定点集的最小面积的包围圆形(圆心,半径)
(2)函数原型:void minEnclosingCircle(InputArray points, Point2f& center, float& radius)
(3)参数说明:输入二维点集,输出圆心,输出半径

(四)用椭圆拟合二维点集:fitEllipse()函数

(1)作用:椭圆拟合二维点集
(2)函数原型:RotatedRect fitEllipse(InputArray points)

(五)逼近多边形曲线:approxPolyDP()函数

(1)作用:用指定精度逼近多边形曲线
(2)函数原型:void approxPolyDP(InputArray curve, OutputArray approxCurve, double epsilon, bool closed)
(3)参数说明:

  • 输入的二维点集
  • 多边形逼近的结果
  • 逼近的精度,为原始曲线和近似曲线间的最大值
  • 取真时,近似的曲线为封闭曲线,取假时不封闭

(六)基础示例程序:创建包围轮廓的矩形边界、圆形边界、椭圆边界

示例代码

#include<opencv2/opencv.hpp>
#include<opencv2/imgproc/imgproc.hpp>
#include<opencv2/highgui/highgui.hpp>
using namespace cv;
using namespace std;

int main()
{
	//初始化变量和随机值
	Mat image(600, 600, CV_8UC3);
	Mat dstImage1, dstImage2, dstImage3, dstImage4;
	RNG& rng = theRNG();

	//循环,按下ESC键程序退出,否则一直更新
	while (1)
	{
		//参数初始化
		int count = rng.uniform(3, 103);//随机生成点的数量
		vector<Point> points;//点值
		//随机生成点坐标
		for (int i = 0; i < count; i++)
		{
			Point point;
			point.x = rng.uniform(image.rows / 4, image.rows * 3 / 4);
			point.y = rng.uniform(image.cols / 4, image.cols * 3 / 4);
			points.push_back(point);
		}
		//绘制出随机颜色的点
		image = Scalar::all(0);
		for (int i = 0; i < count; i++)
		{
			circle(image, points[i], 3, Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 1, 1);

		}

		//【1】对指定点集寻找边界矩形
		Rect rect = boundingRect(Mat(points));
		//绘制边界矩形
		image.copyTo(dstImage1);
		rectangle(dstImage1, rect, Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 2, 1);
		//显示窗口
		imshow("【1】边界矩形窗口", dstImage1);
		
		//【2】对指定点集寻找最小面积的包围矩形
		RotatedRect box1 = minAreaRect(Mat(points));
		Point2f vertex2[4];
		box1.points(vertex2);
		//绘制最小面积的包围矩形
		image.copyTo(dstImage2);
		for (int i = 0; i < 4; i++)
		{
			line(dstImage2, vertex2[i], vertex2[(i + 1) % 4], Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 2, 1);
		}
		//显示窗口
		imshow("【2】最小面积包围矩形窗口", dstImage2);

		//【3】对指定点集寻找最小面积的包围圆形
		Point2f center;
		float radius;
		minEnclosingCircle(Mat(points),center,radius);
		//绘制最小面积的圆形
		image.copyTo(dstImage3);
		circle(dstImage3, center, cvRound(radius), Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 2, 1);
		//显示窗口
		imshow("【3】最小面积包围圆形窗口",dstImage3);

		//【4】椭圆拟合指定点集
		RotatedRect box2 = fitEllipse(Mat(points));
		//绘制椭圆
		image.copyTo(dstImage4);
		ellipse(dstImage4, box2, Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)), 2, 1);
		//显示窗口
		imshow("【4】椭圆形拟合窗口", dstImage4);

		char key = (char)waitKey();
		if(key == 27) break;	
	}
	return 0;
}

运行效果
在这里插入图片描述
在这里插入图片描述

(八)综合示例程序:使用多边形包围轮廓

示例代码

#include<opencv2/opencv.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<opencv2/imgproc/imgproc.hpp>
using namespace cv;
using namespace std;
//定义辅助宏
#define WINDOW_NAME1 "【原始图窗口】"
#define WINDOW_NAME2 "【效果图窗口】"
//全局变量
Mat g_srcImage, g_grayImage;
int g_nThresh = 50;//阈值
int g_nMaxThresh = 255;//阈值最大值
RNG g_rng(12345);//随机数生成器
//全局函数
void on_ContoursChange(int, void*);

int main()
{	
	//【1】载入原图
	g_srcImage = imread("D://lili/Desktop/jpg/opencv/12.jpg");
	if (!g_srcImage.data)
	{
		printf("载入原图失败~!\n");
		return false;
	}
	//【2】创建原始图窗口并显示
	namedWindow(WINDOW_NAME1,WINDOW_AUTOSIZE);
	imshow(WINDOW_NAME1, g_srcImage);
	//【3】得到原图的灰度图像并进行平滑
	cvtColor(g_srcImage, g_grayImage, COLOR_BGR2GRAY);
	blur(g_grayImage, g_grayImage, Size(3, 3));
	//【4】设置滚动条并调用一次回调函数
	createTrackbar("阈值:", WINDOW_NAME1, &g_nThresh, g_nMaxThresh, on_ContoursChange);
	on_ContoursChange(0, 0);

	waitKey(0);
	return 0;
}
void on_ContoursChange(int, void*)
{
	//定义一些参数
	Mat threshold_output;
	vector<vector<Point>> contours;
	vector<Vec4i> hierarchy;

	//使用Threshold检测边缘
	threshold(g_grayImage, threshold_output, g_nThresh, 255, THRESH_BINARY);

	//找出轮廓
	findContours(threshold_output, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));

	//多边形逼近轮廓+获取矩形和圆形边界框
	vector<vector<Point>> contours_poly(contours.size());
	vector<Rect> boundRect(contours.size());
	vector<Point2f> center(contours.size());
	vector<float> radius(contours.size());

	//一个循环,遍历所有部分,进行本程序最核心的操作
	for (int i = 0; i < contours.size(); i++)
	{
		approxPolyDP(Mat(contours[i]), contours_poly[i], 3, true);//用指定精度逼近多边形曲线
		boundRect[i] = boundingRect(Mat(contours_poly[i]));//计算点集的最外面矩形边界
		minEnclosingCircle(contours_poly[i], center[i], radius[i]);//对给定点集寻找最小面积的包围圆形
	}
	//绘制多边形轮廓+包围的矩形框+圆形框
	Mat dstImage = Mat::zeros(threshold_output.size(), CV_8UC3);
	for (int i = 0; i < contours.size(); i++)
	{
		Scalar color = Scalar(g_rng.uniform(0, 255), g_rng.uniform(0, 255), g_rng.uniform(0, 255));//随即设置颜色
		drawContours(dstImage, contours_poly, i, color, 1, 8, vector<Vec4i>(), 0, Point());//绘制轮廓
		rectangle(dstImage, boundRect[i].tl(), boundRect[i].br(), color, 2, 8, 0);//绘制矩形
		circle(dstImage, center[i], (int)radius[i], color, 2, 8, 0);
	}
	//显示效果图窗口
	namedWindow(WINDOW_NAME2,WINDOW_AUTOSIZE);
	imshow(WINDOW_NAME2, dstImage);
}

运行效果
在这里插入图片描述

四、图像的矩

从一幅数字图形中计算出来的矩集,通常描述了该图像形状的全局特征,并提供了大量关于该图像不同类型的几何特性信息,如大小、位置、方向、形状等。图像的一阶矩与形状有关,二阶矩显示曲线围绕直线平均值的扩展程度,三阶矩则是关于平均值的对称性的测量。由二阶矩和三阶矩可以导出一组共7个不变矩。而不变矩是图像的统计特性,满足平移、伸缩、旋转均不变的不变性,在图像识别领域得到了广泛的应用。
计算一个图像的矩,一般由moments、contourArea、arcLength三个函数配合求取。

(一)矩的计算:moments()函数

(1)作用:用于计算多边形和光栅形状的最高达三阶的所有矩。矩用来计算形状的重心、面积、主轴和其他形状特征
(2)函数原型:Moment moments(InputArray array, bool binaryImage=false)
(3)参数说明:

  • 输入参数,可以是光栅图像(单通道,8位或浮点的二维数组)或二维数组(1N或N1)
  • 此参数仅对图像使用,取true,则所有非零像素为1,默认false

(二)计算轮廓面积:contourArea()函数

(1)作用:用于计算整个轮廓或部分轮廓的面积
(2)函数原型:double contourArea(InputArray contour, bool oriented=false)
(3)参数说明:

  • 输入的二维点集
  • 面向区域标识符,若其为true,该函数返回一个带符号的面积值,其正负取决于轮廓方向(顺/逆时针),若其为false,该函数返回面积绝对值。

(三)计算轮廓长度:arcLength()函数

(1)作用:计算封闭轮廓的周长或曲线的长度
(2)函数原型:double arcLength(InputArray curve,bool closed)
(3)参数说明:

  • 输入的二维点集
  • 用于指示曲线是否封闭的标识符,默认closed

(四)综合示例程序:查找和绘制图像轮廓矩

示例代码

#include<opencv2/opencv.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<opencv2/imgproc/imgproc.hpp>
using namespace cv;
using namespace std;

//定义辅助宏
#define WINDOW_NAME1 "【原始图】"
#define WINDOW_NAME2 "【图像轮廓】"
//全局变量
Mat g_srcImage, g_grayImage, g_cannyMat_output;
int g_nThresh = 100;
int g_nMaxThresh = 255;
RNG g_rng(12345);
vector<vector<Point>>g_vContours;
vector<Vec4i>g_vHierarchy;
//全局函数
void on_ThreshChange(int, void*);

int main()
{
	//载入原图
	g_srcImage = imread("D://lili/Desktop/jpg/opencv/12.jpg");
	//把原图转为灰度图并进行平滑
	cvtColor(g_srcImage, g_grayImage, COLOR_RGB2GRAY);
	blur(g_grayImage, g_grayImage, Size(3,3));
	//创建窗口
	namedWindow(WINDOW_NAME1, WINDOW_AUTOSIZE);
	imshow(WINDOW_NAME1, g_srcImage);
	//创建滑动条并初始化
	createTrackbar("阈值:", WINDOW_NAME1, &g_nThresh, g_nMaxThresh, on_ThreshChange);
	on_ThreshChange(0, 0);

	waitKey(0);
	return 0;
}
void on_ThreshChange(int, void*)
{
	//使用Canny检测边缘
	Canny(g_grayImage, g_cannyMat_output, g_nThresh, g_nThresh * 2, 3);
	//找到轮廓
	findContours(g_cannyMat_output, g_vContours, g_vHierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
	
	//计算矩
	vector<Moments> mu(g_vContours.size());
	for (int i = 0; i < g_vContours.size(); i++)
	{
		mu[i] = moments(g_vContours[i], false);
	}
	//计算中心矩
	vector<Point2f> mc(g_vContours.size());
	for (int i = 0; i < g_vContours.size(); i++)
	{
		mc[i] = Point2f(static_cast<float>(mu[i].m10 / mu[i].m00), static_cast<float>(mu[i].m01 / mu[i].m00));
	}

	//绘制轮廓
	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());//绘制外层和内层轮廓
		circle(drawing, mc[i], 4, color, -1, 8, 0);//绘制圆
	}
	//显示窗口
	namedWindow(WINDOW_NAME2,WINDOW_AUTOSIZE);
	imshow(WINDOW_NAME2, drawing);

	//通过m00计算轮廓面积和OpenCV函数比较
	printf("\t 输出内容:面积和轮廓长度\n");
	for (int i = 0; i < g_vContours.size(); i++)
	{
		printf(">通过m00计算出轮廓[%d]的面积:(M_00) = %.2f \n", i, mu[i].m00);
		printf(" OpenCV函数计算出的面积 = %.2f, 长度:%.2f \n\n", contourArea(g_vContours[i]), arcLength(g_vContours[i], true));
	}
}

运行效果
在这里插入图片描述

五、分水岭算法

  • 基于拓扑理论的数学形态学的分割方法。
  • 基本思想:把图像看作测地学上的拓扑地貌,图像中每一点像素的灰度值表示该点的海拔高度,每一个局部极小值及其影响区域称为集水盆,分水岭变换得到的是输入图像的集水盆图像,集水盆边界形成分水岭(输入图像的极大值点)。
  • 计算过程:一个迭代标注过程,包括排序过程和淹没过程,先对每个像素的灰度级进行从低到高的排序,然后在从低到高实现淹没的过程中,对每一个局部最小值在h阶高度的影响域采用先进先出(FIFO)结构进行判断及标注。

(一)实现分水岭算法:watershed()函数

基于标记的分割算法,在把图像传给函数之前,需要大致标记出图像中期望进行分割的区域,每个区域被标记为像素值1、2、3等,表示成为一个或多个连接组件,标记的值可使用findContours()函数和drawContours()函数由二进制的掩码检索出来,函数输出中,每一个标记中的像素被设置为标记的值,区域间的值被设置为-1。

void watershed(InputArray image,InputOutputArray markers)
  • 输入图像,8位三通道彩色图像
  • 函数调用后运算结果,输入/输出32位单通道图像的标记结果

(二)综合示例程序:分水岭算法

有些难以理解。但是没关系,了解一下原理,毕竟这是传统的图像分割方法,现在已经有了更好的分割方法了。
示例代码

/*
  程序说明:鼠标大致标记出图像中期望进行分割的区域
           键盘按键【1】启动分水岭算法
           按键【2】恢复原始图重新标记
*/
#include<opencv2/opencv.hpp>
#include<opencv2/imgproc/imgproc.hpp>
#include<opencv2/highgui/highgui.hpp>
#include<iostream>
using namespace cv;
using namespace std;
//定义辅助宏
#define WINDOW_NAME1 "【原始图窗口】"
#define WINDOW_NAME2 "【恢复原始图窗口】"
#define WINDOW_NAME3 "【分水岭变换窗口】"
//全局变量
Mat g_srcImage, g_maskImage;
Point prevPt(-1, -1);
//全局函数
static void on_Mouse(int event, int x, int y, int flags, void*);
static void ShowHelpText();

int main()
{
	//【0】显示帮助信息
	ShowHelpText();
	
	//【1】载入原图并显示
	g_srcImage = imread("D://lili/Desktop/jpg/opencv/9.jpg");
	if (!g_srcImage.data)
	{ 
		printf("载入原图像失败\n");
		return false;
	}
	imshow(WINDOW_NAME1, g_srcImage);
	//【2】初始化掩模和灰度图
	Mat srcImage, grayImage;
	g_srcImage.copyTo(srcImage);
	cvtColor(srcImage, g_maskImage, COLOR_BGR2GRAY);
	cvtColor(g_maskImage, grayImage, COLOR_GRAY2BGR);
	g_maskImage = Scalar::all(0);
	//【3】设置鼠标回调函数
	setMouseCallback(WINDOW_NAME1, on_Mouse, 0);

	//【4】轮询按键,进行处理
	while (1)
	{
		//获取键值
		int c = waitKey(0);
		//按键值为ESC时,退出程序
		if ((char)c == 27)
			break;
		//按键为2时,恢复原图,使g_maskImage和g_srcImage可重新标记
		if ((char)c == '2')
		{
			g_maskImage = Scalar::all(0);
			srcImage.copyTo(g_srcImage);
			imshow(WINDOW_NAME2, g_srcImage);
		}
		//按键值为1时,进行处理
		if ((char)c == '1')
		{
			//定义一些参数
			int i, j;
			int compCount = 0;  //记录轮廓数
			vector<vector<Point>>contours;
			vector<Vec4i> hierarchy;
			//寻找轮廓
			findContours(g_maskImage, contours, hierarchy, CV_RETR_CCOMP, CHAIN_APPROX_SIMPLE);
			//轮廓为空时的处理
			if (contours.empty())
				continue;
			//复制掩膜
			Mat maskImage(g_maskImage.size(), CV_32S);
			maskImage = Scalar::all(0);

			//循环绘制轮廓
			for (int index = 0; index >= 0; index = hierarchy[index][0], compCount++)
			{
				drawContours(maskImage, contours, index, Scalar::all(compCount + 1), -1, 8, hierarchy, INT_MAX);
			}

			//为每一个轮廓生成一个随机颜色
			vector<Vec3b>colorTab;
			for (int i = 0; i < compCount; i++)
			{
				int b = theRNG().uniform(0, 255);
				int g = theRNG().uniform(0, 255);
				int r = theRNG().uniform(0, 255);
				colorTab.push_back(Vec3b((uchar)b, (uchar)g, (uchar)r));
			}
			//进行分水岭算法并计算处理时间输出到窗口中
			double dTime = (double)getTickCount();
			watershed(srcImage, maskImage);
			dTime = (double)getTickCount() - dTime;
			printf("\t 处理时间 = %gms\n", dTime*1000. / getTickFrequency());
			
			//双层循环,将分水岭图像遍历存入watershedImage中
			Mat watershedImage(maskImage.size(), CV_8UC3);
			for (i = 0; i < maskImage.rows; i++)
				for (j = 0; j < maskImage.cols; j++)
				{
					int index = maskImage.at<int>(i, j);
					if (index == -1)
						watershedImage.at<Vec3b>(i, j) = Vec3b(255, 255, 255);//maskImage中区域间值像素为-1,将对应watershed图像像素设置为白色(255,255,255)
					else if (index <= 0 || index > compCount)
						watershedImage.at<Vec3b>(i, j) = Vec3b(0, 0, 0);      //maskImage中非标记区域,将对应watershed图像像素设置为黑色(0,0,0)
					else
						watershedImage.at<Vec3b>(i, j) = colorTab[index - 1]; //maskImage中标记区域,将对应watershed图像像素设置为之前随机出的颜色colorTab
				}
			//混合灰度图和分水岭效果图并显示最终的窗口
			watershedImage = watershedImage * 0.5 + grayImage * 0.5;
			imshow(WINDOW_NAME3, watershedImage);
		}
	}
	return 0;
}

//鼠标消息回调函数
void on_Mouse(int event, int x, int y, int flags, void*)
{
	//处理鼠标不在窗口中的情况
	if (x < 0 || x >= g_srcImage.cols || y < 0 || y >= g_srcImage.rows)
		return;

	//处理鼠标左键相关消息
	if (event == EVENT_LBUTTONUP || !(flags & EVENT_FLAG_LBUTTON)) //松开左键
		prevPt = Point(-1, -1);
	else if (event == EVENT_LBUTTONDOWN)//按下左键
		prevPt = Point(x, y);           //记录鼠标按下时的位置,作为白色线条的起始点

	//鼠标左键呈按下状态并移动,绘制出白色线条
	else if (event == EVENT_MOUSEMOVE && (flags & EVENT_FLAG_LBUTTON))
	{
		Point pt(x, y);
		if (prevPt.x < 0)  //如果白线起始点范围溢出,则白线起始点为当前点pt
			prevPt = pt;
		line(g_maskImage, prevPt, pt, Scalar::all(255), 2, 8, 0);//画白线
		line(g_srcImage, prevPt, pt, Scalar::all(255), 2, 8, 0); //画白线
		prevPt = pt;
		imshow(WINDOW_NAME1, g_srcImage);
	}
}
static void ShowHelpText()
{
	printf("\n\n\t欢迎来到【分水岭算法】示例程序~\n");
	printf("\n\t请先用鼠标在图片窗口中标记出大致的区域\n");
	printf("\n\t然后再按键【1】启动分水岭算法\n");
	printf("\n\t按键操作说明:\n");
	printf("\t\t\t键盘按键【1】--运行分水岭分割算法\n");
	printf("\t\t\t键盘按键【2】--恢复原始图\n");
	printf("\t\t\t键盘按键【ESC】--退出程序\n");
}

运行效果
在这里插入图片描述

六、图像修补

基本思想:
  利用已经被破坏区域的边缘,即边缘的颜色和结构,繁殖和混合到损坏的图像中,达到图像修补的目的。

(一)实现图像修补:inpaint()函数

用来从扫描的照片中清除灰尘和划痕,从静态图像或视频中去除不需要的物体

void inpaint(InputArray src, InputArray inpaintMask, OutputArray dst, double inpaintRadius, int flags)

(1)输入图像,8位单通道或三通道
(2)修复掩模,8位单通道,其中的非零像素表示需要修补区域
(3)修补后图像
(4)需要修补的每个点的圆形区域,为修复算法的参考半径
(5)修补方法的标识符,可取值:

  • INPAINT_NS:基于Navier-Stokes方程的方法
  • INPAINT_TELEA:Alexandru Telea方法

(二)综合示例程序:图像修补

示例代码

/*
	程序说明:鼠标绘制白色线条破坏原图像图像
			 键盘按键【1】启动进行图像修复
			 按键【2】恢复原始图像
*/

#include<opencv2/opencv.hpp>
#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, g_inpaintMask;
Point previousPoint(-1, -1); //原来的点坐标
//全局函数
static void on_Mouse(int event, int x, int y, int flags, void*);
static void ShowHelpText();

int main()
{
	//显示帮助文字
	ShowHelpText();

	//载入原图并显示
	Mat srcImage = imread("D://lili/Desktop/jpg/opencv/9.jpg");
	if (!srcImage.data)
	{
		printf("载入原图失败\n");
		return false;
	}
	imshow(WINDOW_NAME1, srcImage);
	//初始化掩模
	g_srcImage = srcImage.clone();
	g_inpaintMask = Mat::zeros(g_srcImage.size(), CV_8U);
	//设置鼠标回调消息
	setMouseCallback(WINDOW_NAME1, on_Mouse, 0);
	//按键轮询
	while (1)
	{
		//获取按键键值
		char c = (char)waitKey();
		//键值为ESC,程序退出
		if (c == 27) break;
		//键值为2,恢复原始图像
		if (c == '2')
		{
			g_inpaintMask = Scalar::all(0);
			srcImage.copyTo(g_srcImage);
			imshow(WINDOW_NAME1, g_srcImage);
		}
		//键值为1,进行图像修复
		if (c == '1')
		{
			Mat inpaintedImage;
			inpaint(g_srcImage, g_inpaintMask, inpaintedImage, 3, INPAINT_TELEA);
			imshow(WINDOW_NAME2, inpaintedImage);
		}
	}
	return 0;
}
//鼠标回调函数
static void on_Mouse(int event, int x, int y, int flags, void*)
{
	//鼠标左键弹起消息
	if (event == EVENT_LBUTTONUP || !(flags & EVENT_FLAG_LBUTTON))
	{
		previousPoint = Point(-1, -1);
	}
	//鼠标左键按下消息
	else if (event == EVENT_LBUTTONDOWN)
	{
		previousPoint = Point(x, y);
	}
	//鼠标按下状态并移动,进行绘制
	else if (event == EVENT_MOUSEMOVE && (flags & EVENT_FLAG_LBUTTON))
	{
		Point pt(x, y);
		if (previousPoint.x < 0)
		{
			previousPoint = pt;
		}
		line(g_inpaintMask, previousPoint, pt, Scalar::all(255), 5, 8, 0);
		line(g_srcImage, previousPoint, pt, Scalar::all(255), 5, 8, 0);
		previousPoint = pt;
		imshow(WINDOW_NAME1, g_srcImage);
	}
}
static void ShowHelpText()
{
	printf("\n\t欢迎来到【图像修复】示例程序~\n");
	printf("\n\t请再进行图像修复操作之前,在【原始图】窗口中进行适量的绘制\n");
	printf("\n\t按键操作说明:\n");
	printf("\t\t\t键盘按键【1】--进行图像修复\n");
	printf("\t\t\t键盘按键【2】--恢复原始图\n");
	printf("\t\t\t键盘按键【ESC】--退出程序\n");
}

运行效果
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/685928.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

FFmpeg 播放器实现音视频同步的几种方式

我们基于 FFmpeg 利用 OpenGL ES 和 OpenSL ES 分别实现了对解码后视频和音频的渲染&#xff0c;本文将实现播放器的最后一个重要功能&#xff1a;音视频同步。 老人们经常说&#xff0c;播放器对音频和视频的播放没有绝对的静态的同步&#xff0c;只有相对的动态的同步&#…

深度学习管理工具

1. mlflow https://zhuanlan.zhihu.com/p/363673389 官方文档&#xff1a; https://www.mlflow.org/docs/latest/index.html https://zhuanlan.zhihu.com/p/67173051 在跟踪实验时&#xff0c;通过运行 mlflow ui 后在 http&#xff1a;// localhost&#xff1a;5000 中进行…

ranger,hive,hdfs的三者的权限管理

ranger&#xff0c;hive&#xff0c;hdfs的三者的权限管理 情况一&#xff1a;连接datagrip 用户在hdfs上的权限 可以看出只给了用户write权限&#xff0c;尝试登录xwq用户&#xff0c;在datagrip上登录成功 经过实验验证&#xff1a;要想使用datagrip或者hive-cli登录hive…

C++11【一】

文章目录 一、C11简介二、右值引用三、可变参数模板四、 empacle_back(移动构造/赋值)五、简单特性 一、C11简介 C11是C编程语言的一个版本&#xff0c;于2011年发布。C11引入了很多新特性&#xff0c;比如&#xff1a;类型推导(auto关键字)、Lambda表达式、线程库、列表初始化…

电源浪涌保护器加装后备保护器的作用

随着现代化技术的发展&#xff0c;信息化系统集成度越来越高。设备敏感度高&#xff0c;抗冲击能力低&#xff0c;极易受到电涌脉冲的危害。电涌保护器&#xff08;SPD&#xff09;作为专业的防电涌产品&#xff0c;应用越来越广&#xff0c;SPD的应用涉及很多的专业知识&#…

孩子创新思维秘籍

头脑风暴最重要的原则之一就七个字&#xff0c; 要数量&#xff0c;不要质量。 意思就是说&#xff0c;你尽可能的去胡思乱想&#xff0c;胡说八道&#xff0c;天马行空就对了。 我们从这一大堆古灵精怪的玩意儿里面去筛选那些真正有价值的东西&#xff0c;可以拓展的方案。 然…

【技术选型】Redis的几种集群方案、及优缺点对比

文章目录 背景一、主从模式二、哨兵模式三、Redis Cluster四、各大厂的Redis集群方案客户端分片代理分片Codis 五、Redis集群方案解决方案六、Redis集群方案原理总结 背景 在服务开发中&#xff0c;单机都会存在单点故障的问题&#xff0c;及服务部署在一台服务器上&#xff0…

GCP学习笔记(二)——大数据和机器学习

文章目录 一、数据读取和处理1.Pub/Sub代码实践PublishingSubscribing 2. Dataflow使用Python搭建Pipeline 3. Dataproc4. Cloud Data Fusion5. 其他工具Cloud Composer &#xff08;Apache Airflow&#xff09;Cloud Scheduler 二、可视化与分析1. Looker2. Looker Studio3. B…

【数据结构与算法C++实现】2、二分查找与简单递归

原视频为左程云的B站教学 文章目录 1 二分法1.1 在有序数组中查找特定元素1.2 在一个有序数组中查找>某个数的最左侧的位置1.3 在一个有序数组中查找<某个数最右侧位置1.4 局部最小值问题&#xff08;无序数组使用二分法的案例&#xff09; 2 简单的递归思想 1 二分法 …

在AI热潮中,过早的卖掉Datadog股票是个非常错误的决定

来源&#xff1a;猛兽财经 作者&#xff1a;猛兽财经 Datadog股价 Datadog&#xff08;DDOG&#xff09;目前的股价比其历史高点低了50%左右&#xff0c;比近期低点高了50%左右。 猛兽财经上次关注Datadog是在4月份&#xff0c;当时由于该股的增长前景已经恶化&#xff0c;所以…

windows电脑hbuilderx打包iOS app及上架app store教程

ios应用&#xff0c;无法像安卓应用一样&#xff0c;上传到自己的服务器让互联网用户下载进行安装&#xff0c;所以需要将生成的app上传到app store&#xff0c;然后用户到app store安装app。 由于官网的教程是使用mac电脑生成证书和上架的&#xff0c;但是很多使用hbuilderx打…

实践指南 | 风控引擎快速接入不同数据源的操作说明

随着互联网垂直电商、消费金融等领域的快速崛起&#xff0c;用户及互联网、金融平台受到欺诈的风险也急剧增加。网络黑灰产已形成完整的、成熟的产业链&#xff0c;每年千亿级别的投入规模&#xff0c;超过1000万的“从业者”&#xff0c;其专业度也高于大多数技术人员&#xf…

Templates 虽然工具将近被淘汰,但依然会有一些场景会被使用-eclipse 格式化注释

Templates 是一款-eclipse 格式化注释模板。虽然工具将近被淘汰&#xff0c;但依然会有一些场景会被使用&#xff0c;今天就来分享一下自己用过的一款&#xff0c;已经去掉不常用的&#xff0c;保留必要的模板内容。 设置方法如下&#xff1a; 设置Code Templates&#xff0c…

Java-定时任务

文章目录 补充&#xff1a;cron表达式基本知识方式一&#xff1a;使用sleep方法方式二&#xff1a;JDK Timer和TimerTask方式三&#xff1a;JDK ScheduledExecutorService方式四&#xff1a; Spring Task 中 的 Scheduler方法五、Quartz框架方式六&#xff1a;XXL-JOB将xxl-job…

canvas实现简易画板

效果图如下&#xff1a; 实现功能&#xff1a; 1、改变画笔粗细 2、保存签名实现下载功能 3、使用橡皮擦功能 4、清空画布 5、改变画笔颜色 实现代码如下 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta …

ChatGPT新功能曝光:可记住用户信息、上传文件和工作区

&#x1f989; AI新闻 &#x1f680; ChatGPT新功能曝光&#xff1a;可记住用户信息、上传文件和工作区 摘要&#xff1a;一张神秘截图曝光了ChatGPT新功能&#xff0c;包括可记住用户信息的"My profile"、上传和管理文件的"My files"以及可以让AI使用不…

从Wi-Fi,蓝牙,到4G,5G,到卫星网络,频谱共享已无处不在

在智能手机像牙刷一样普及的今天&#xff0c;频谱共享&#xff0c;成为了近些年通信界的一个热词儿。频谱因为通信的重要而变得越发重要&#xff0c;又因为频谱是一种稀缺资产&#xff0c;而使用需求又在日益飞速地增长&#xff0c;所以成为重中之重。智能手机、物联网、军事和…

【QT】枚举常用宏到底有什么作用?(Q_ENUM,Q_FLAG,Q_DECLARE_FLAGS,Q_DECLARE_OPERATORS_FOR_FLAGS)

目录 1. Q_ENUM宏 与 QMetaEnum类1.1 Q_ENUM宏的作用1.2 使用Q_ENUM注意的问题1.3 在写有关枚举的代码时&#xff0c;我们可能遇到这种情况&#xff1a;需要用到枚举的字符串&#xff0c;该怎么办&#xff1f;1.4 下面通过一段简单的代码来说明Q_ENUM的作用 2. Q_FLAG宏2.1 Q_F…

satellite: 利用TLE动态计算并实时显示多颗卫星的位置及轨迹

本示例的目的是介绍演示如何在vue+satellite项目中利用两行根数动态地计算,并显示多个卫星的位置及轨迹。每秒钟更新一下卫星的位置和角度,加载当前时间到固定时间(如720分钟后)的一段轨迹。 直接复制下面的 vue+openlayers源示例代码,操作2分钟即可运行实现效果 文章目…

DDD领域驱动设计基本理解

DDD是一种软件设计思想和方法论&#xff0c;以领域为核心构建软件设计体系&#xff0c;将业务模型抽象成领域模型进行拆解和封装。本文简要介绍DDD的基本概念和常用的分层设计架构&#xff0c;并结合业务场景进行领域驱动设计的实战分析&#xff0c;以加深理解。 1、DDD领域驱动…