基于OpenCV的简易实时手势识别

  • 1.基本信息介绍
    • 1.1实验步骤
    • 1.2效果展示
  • 2.肤色检测+二值化+开运算+高斯模糊
    • 2.1 flip()函数原型
    • 2.2cvtColor()函数原型
    • 2.3split()函数原型
    • 2.4GaussianBlur()函数原型
    • 2.5Code
  • 3.连通空心部分+腐蚀
    • 3.1 floodFill()函数原型
    • 3.2 morphologyEx()函数原型
    • 3.3Code
  • 4.多边形拟合曲线
    • 4.1approxPolyDP()函数原型
    • 4.2Code
  • 5.凸包检测+重心+ 鼠标操作
    • 5.1convexHull()函数原型
    • 5.2moments()函数原型
    • 5.3Mouse_event()函数原型
    • 5.5 Code
  • 6.主函数 摄像头调用
    • 6.1摄像头调用
    • 6.2 Code
  • 7.代码中的其他API函数
    • 7.1 getStructuringElement()函数原型
    • 7.2 findContours()函数原型
    • 7.3 circle()函数原型
    • 7.4 line()函数原型
    • 7.5 namedWindow()函数
    • 7.6 imshow()函数
  • 8.参考文献

1.基本信息介绍

这是我大一寒假时写着玩的,非常简陋。基于凸包检测,所以实际上是计算指尖数量判断1~5的手势。又为1 ~3手势赋了控制鼠标操作的功能(但不能移动鼠标,而且因为手势识别不太准确所以这个功能实现得很废/doge)。(才疏学浅,希望有生之年能写个更好的
版本信息:Visual Studio2015 OpenCV4.1.1
语言:C/C++
(至于为什么不用python,现在当事人也很后悔

1.1实验步骤

(1)图像捕获
直接调用笔记本内置摄像头,使图像绕y轴对称翻转,得到内置(前置)摄像头所拍摄的视频画面的镜像画面,从而得到以操作者为第一视角的正向画面。

(2)肤色检测
先将图像由RGB空间转换至YCrCb空间。
然后将图像成Y(像素的亮度)、Cb(红色分量与亮度的信号差值)、Cr(蓝色分量与亮度的信号差值)三个单通道图像。
再提取Cb、Cr两通道的图像,进行数值判断,满足Cr>133 && Cr<173 && Cb>77 && Cb<127的点即判断为肤色部分。
最后将完成肤色切割的图像进行二值化。

(3)图像预处理
本实验先使用开运算(即先腐蚀后膨胀)对二值化后的手掌图像进行处理,去除图中的小孤立点,消除较小连通域,保留较大连通域,在不明显改变较大连通域面积的同时平滑连通域的边界,是手掌轮廓更明显,为之后的漫水填充做准备。
然后进行高斯滤波,从而消除图像上的高斯噪声。
再通过漫水填充算法,将手掌中因光线角度等因素在肤色检测中缺失的部分填充。
最后图像腐蚀处理漫水填充后的图像,将细小的噪声去除的同时,将图像主要区域的面积缩小。为之后的多边形拟合曲线求得图像近似轮廓做准备。

(4)指尖检测
本实验先用多边形逼近手部轮廓,求得近似轮廓。
再使用凸包检测函数对手部轮廓进一步进行多边形逼近,进而获得一个凸多边形。找到重心位置,通过比较凸包的顶点与重心的y轴坐标,去除纵坐标小于重心纵坐标的顶点,保留纵坐标大于重心的凸包的顶点,再规定凸点间距离范围以消除由同一个指尖产生的多个凸包顶点,得到指尖数量。

(5)模拟鼠标
最后通过得到的指尖数量,控制鼠标操作。
当指尖数量=1时,在图像重心处显示“Left”,同时执行鼠标左键单击功能。
当指尖数量=2时,在图像重心处显示“Double click”,同时执行鼠标左键双击功能。
当指尖数量=3时,在图像重心处显示“Right”,同时执行鼠标右键单击功能。

1.2效果展示

#include<opencv2\opencv.hpp>
#include<iostream>
#include<vector>
#include<algorithm>
#include<math.h>
#include<Windows.h>using namespace std;
using namespace cv;/*介绍基本信息*/
void Introduce()
{cout << "\n----------------------------------------------------------------------------";cout << "\n功能:以手势代替鼠标进行左右键点击";cout << "\n版本信息:Visual Studio2015  OpenCV4.1.1";cout << "\n-------------------------------------指令集---------------------------------";cout << "\n手势1:单击鼠标左键Left";cout << "\n手势2:双击鼠标左键Double click";cout << "\n手势3:单击鼠标右键Right";cout << "\n----------------------------------------------------------------------------\n";
}

2.肤色检测+二值化+开运算+高斯模糊

2.1 flip()函数原型

本实验通过使图像绕y轴对称翻转,得到内置(前置)摄像头所拍摄的视频画面的镜像画面,从而得到以操作者为第一视角的正向画面。

flip()函数原型
flip(   InputArray      src,
OutputArray dst,
Int             flipCode
)

①src:输入图像。
②dst:输出图像,与src具有相同的大小、数据类型及通道数。
③flipCode:翻转方式标志。数值大于0表示绕y轴翻转;数值等于0表示绕x轴翻转;数值小于0,表示绕两个轴翻转。

2.2cvtColor()函数原型

本实验中肤色检测步骤如下:
①通过颜色模型转换函数cvtColor()函数将图像由RGB空间转换至YCrCb空间。
②通过多通道分离函数split()将图像成Y(像素的亮度)、Cb(红色分量与亮度的信号差值)、Cr(蓝色分量与亮度的信号差值)三个单通道图像。
③提取Cb、Cr两通道的图像,进行数值判断,满足Cr>133 && Cr<173 && Cb>77 && Cb<127的点即判断为肤色部分。
④将完成肤色切割的图像进行二值化。

cvtColor()函数原型
cvtColor(   InputArray      src,
OutputArray dst,
int     code,
int     dstCn   =0
)

①src:待转换颜色模型的原始图像。
②dst:转换颜色模型后的目标图像。
③code:颜色空间转换的标志。本实验使用的标志参数为。
④dstCn:目标图像中的通道数。若参数为0,则从src和代码中自动导出通道数。本实验中使用默认参数。

2.3split()函数原型

split()函数原型
split(  const       Mat& src,
Mat *       mvbegin
)
split(  InputArray              m,
OutputArrayOfArrays mv
)

①src:待分离的多通道图像。
②mvbegin:分离后的单通道图像,为数组形式,数组大小需要与图像的通道数一致。
③m:待分离的多通道图像。
④mv:分离后的单通道图像,为向量(vector)形式。

2.4GaussianBlur()函数原型

在图像采集的众多过程中都容易引用高斯噪声。高斯滤波器考虑了像素滤波器中心距离的影响,以滤波器中心位置为高斯分布的均值,根据高斯分布公式和每个像素离中心位置的距离计算出滤波器内每个位置的数值,从而形成一个高斯滤波器。在将高斯滤波器与图像之间进行滤波操作,进而实现对图像的高斯滤波。
本实验使用GaussianBlur()函数进行高斯滤波。

GaussianBlur()函数原型
GaussianBlur(   InputArray      src,
OutputArray dst,
Size            ksize,
double          sigmaX,
double          sigmaY=0,
int             borderType=BORDER_DEFAULT(默认参数)
)

①src:待高斯滤波的图像,图像的数据类型必须为CV_8U、CV_16U、CV_16S、CV_32F或CV_64F,通道数目任意。
②dst:输出图像,与src尺寸、通道数、数据类型都相同。
③ksize:高斯滤波器的尺寸。滤波器必须是政奇数。如果尺寸为0,则由标准偏差计算尺寸。
④sigmaX:X轴方向的高斯滤波器标准偏差。
⑤sigmaY:Y轴方向的高斯滤波器标准偏差。如果输入量为0,则将其设置为等于sigmaX;如果两个轴的标准差都为0,则根据输入的高斯滤波器尺寸计算标准偏差。
⑥borderType:像素外推法选择标志。(边界外推方法标志见下表)

方法标志参数 简记 作用
BORDER_CONSTANT 0 用特定值填充
BORDER_REPLICATE 1 两端复制填充
BORDER_REFLECT 2 倒序填充
BORDER_WRAP 3 正序填充
BORDER_REFLECT_101 4 不包含边界值的倒序填充
BORDER_TRANSPARENT 5 随机填充
BORDER_REFLECT101 4 同BORDER_REFLECT_101
BORDER_DEFAULT 4 同BORDER_DEFAULLT
BORDER_ISOLATED 16 不关心感兴趣区域之外的部

2.5Code

/*基于YCrCb空间的肤色检测+二值化+开运算+高斯模糊*/
Mat skin(Mat&ImageIn)
{Mat Image_y;flip(ImageIn, Image_y, 1);//将图像沿y轴翻转,即镜像namedWindow("前置摄像头", WINDOW_NORMAL | WINDOW_KEEPRATIO);imshow("前置摄像头", Image_y);Mat Image = Image_y.clone();//用clone()函数复制图像Mat YCrCb_Image;cvtColor(Image, YCrCb_Image, COLOR_BGR2YCrCb);vector<Mat>Y_Cr_Cb;split(YCrCb_Image, Y_Cr_Cb);Mat CR = Y_Cr_Cb[1];Mat CB = Y_Cr_Cb[2];Mat ImageOut = Mat::zeros(Image.size(), CV_8UC1);//zeros():构建一个全为0的矩阵,即创建一个全黑的图片//Cr>133 && Cr<173 && Cb>77 && Cb<127for (int i = 0; i < Image.rows; i++){for (int j = 0; j < Image.cols; j++){if (CR.at<uchar>(i, j) >= 133 && CR.at<uchar>(i, j) <= 173 && CB.at<uchar>(i, j) >= 77 && CB.at<uchar>(i, j) <= 127){ImageOut.at<uchar>(i, j) = 255;}}}Mat kernel = getStructuringElement(MORPH_RECT, Size(3, 3));//结构元素 表示内核为一个3*3的矩形morphologyEx(ImageOut, ImageOut, MORPH_OPEN, kernel);//使用morphologyEx()函数进行开运算GaussianBlur(ImageOut, ImageOut, Size(3, 3), 5);return ImageOut;
}

3.连通空心部分+腐蚀

3.1 floodFill()函数原型

漫水填充法是根据像素灰度值之间的差值寻找相同区域以实现分割。本实验通过floodFill()函数,将手掌中因光线角度等因素在肤色检测中缺失的部分填充。
漫水填充法主要步骤如下:
①选择种子点。
②以种子为中心,判断4-领域或者8-领域的像素值与中子点像素值的差值,将差值小于阈值的像素点添加进区域内。
③将新加入的像素点作为新的种子点,反复执行第二步,直到没有新的像素点被添加进该区域为止。

floodFill()函数原型
floodFill(  InputOutputArray        image,
InputOutputArray        mask,
Point                   seedPoint,
Scalar                  newVal,
Rect                    *rect=0,
Scalar                  loDiff = Scalar(),
Scalar                  upDiff = Scalar(),
int                     flags = 4
)

①image:输入及输出图像,可以为CV_8U或CV_32F数据类型的单通道或三通道图像。
②mask:掩码矩阵,尺寸比输入图像宽和高各大2的单通道图像,用于标记漫水填充的区域
③seedPoint:种子点,可以为图像范围内任意一点。
④newVal:归入种子点区域内像素点的新像素值,该值会直接作用在原图中。
⑤rect:种子点漫水填充区域的最小矩形边界,默认值为0,表示不输出边界。
⑥loDiff:添加进种子点区域条件的下界差值,当邻域某像素点的像素值域与种子点像素值的差值大于该值时,该像素点被添加进种子所在的区域。
⑦upDiff:添加进种子点区域条件的上界差值,当种子点像素值与邻域某像素点的像素值的差值小于该值时,该像素点被添加进种子点所在的区域。
⑧flags:漫水填充法的操作标志,由3部分构成,分别表示邻域种类、掩码矩阵中被填充像素点的像素值和填充算法的规则,填充算法可选的标志如下表

操作标志参数 简记 含义
FLOODFILL_FIXED_RANGE 1<<16 如果设置该参数,那么仅考虑当前像素值与初始种子点像素之间的差值,否则考虑新种子点像素值与当前像素值之间的差异,即范围是否浮动的标志
FLOODFILL_MASK_ONLY 1<<17 如果设置,那么该函数不会更改原始图像,即忽略第四个参数newVal,只生成掩码矩阵

3.2 morphologyEx()函数原型

本实验中使用开运算处理肤色检测处理后的图像,去除图中的噪声,消除较小连通域,保留较大连通域,并且能够在不明显改变较大连通域面积的同时平滑连通域的边界,为之后的漫水填充做准备。
本实验中还使用了图像腐蚀处理漫水填充后的图像,将细小的噪声去除的同时,将图像主要区域的面积缩小。为之后的多边形拟合曲线求得图像近似轮廓做准备。
OpenCV4中提供了图像腐蚀和膨胀运算不同组合形式的morphologyEx()函数。

morphologyEx()函数原型
MorphologyEx(   InputArray      src,
OutoutArray dst,
int             op,
InputArray      kernel,
Point   anchor = point(-1,-1),
int     iterations = 1,
Int     borderType = BORDER_CONSTANT,
Const Scalar & borderValue = morphologyDefaultBorderValue()
)

src:输入图像
dst:形态学操作后的输出图像
op:形态学操作类型的标志,可选择的标志及其函数如下表所示
kernel:结构元素,可以自己生成,也可以用getStructuringElement()函数生成
anchor:中心点在结构元素中的位置,默认参数为结构元素的集合中心点。
iterations:处理的次数
borderType:像素外推法选择标志
borderValue:使用边界不变外推法时的边界值。

形态学操作类型标志参数 简记 含义
MORPH_ERODE 0 图像腐蚀
MORPH_DILATE 1 图像膨胀
MORPH_OPEN 2 开运算
MORPH_CLOSE 3 闭运算
MORPH_GRANDIENT 4 形态学梯度
MORPH_TOPHAT 5 顶帽运算
MORPH_BLACKHAT 6 黑帽运算
MORPH_HITMISS 7 击中击不中运算

3.3Code

/*连通空心部分+腐蚀*/
Mat Floodfill(Mat&Img_src)
{Size f_size = Img_src.size();Mat image = Mat::zeros(f_size.height + 2, f_size.width + 2, CV_8UC1);Img_src.copyTo(image(Range(1, f_size.height + 1), Range(1, f_size.width + 1)));floodFill(image, Point(0, 0), Scalar(255));Mat cutImg, Img_dst;image(Range(1, f_size.height + 1), Range(1, f_size.width + 1)).copyTo(cutImg);Img_dst = Img_src | (~cutImg);Mat kernel1 = getStructuringElement(MORPH_RECT, Size(10, 10));//结构元素 表示内核为一个10*10的矩形morphologyEx(Img_dst,Img_dst, MORPH_ERODE, kernel1);//使用morphologyEx()函数进行腐蚀运算return Img_dst;
}

4.多边形拟合曲线

4.1approxPolyDP()函数原型

本实验通过approxPolyDP()函数对图像进行处理,用多边形逼近手部轮廓,求得近似轮廓,为之后的凸包检测做准备。

approxPolyDP()函数原型
approxPolyDP(   InputArray      curve,
OutputArray approxCurve,
double      epsilon,
bool            closed
)

①curve:输入轮廓像素点。
②approxCurve:多边形逼近结果,以多边形顶点坐标的形式给出。
③epsilon:逼近的精度,即原始曲线和逼近曲线之间的最大距离。
④closed:逼近曲线是否为封闭曲线的标志,true表示封闭。

4.2Code

/*计算两点间距离*/
double distance(Point a, Point b)
{double distance = sqrt(abs((a.x - b.x)*(a.x - a.x) + (a.y - b.y)*(a.y - b.y)));return distance;
}/*将坐标点连接成封闭图形*/
void draw(Mat Img1, Mat Img2)
{for (int i = 0;i < Img1.rows;i++){if (i == Img1.rows - 1){Vec2i point1 = Img1.at<Vec2i>(i);Vec2i point2 = Img1.at<Vec2i>(0);line(Img2, point1, point2, Scalar(255, 255, 255), 2, 8, 0);break;}Vec2i point1 = Img1.at<Vec2i>(i);Vec2i point2 = Img1.at<Vec2i>(i + 1);line(Img2, point1, point2, Scalar(255, 255, 255), 5, 8, 0);}
}/*多边形拟合曲线绘制近似轮廓*/
Mat approx(Mat&Img_src)
{Mat Img_dst = Mat::zeros(Img_src.size(), CV_8UC1);vector<vector<Point>>contours;vector<Vec4i>hierarchy;findContours(Img_src, contours, hierarchy, 0, 2, Point());for (int t = 0;t < contours.size();t++){Mat app;approxPolyDP(contours[t], app, 15, true);draw(app, Img_dst);}return Img_dst;
}

5.凸包检测+重心+ 鼠标操作

5.1convexHull()函数原型

在图形学中,将二维平面上的点集最外层的点连接起来构成的凸多边形称为凸包。
本实验通过用于物体凸包检测的convexHull()函数,对手部轮廓进一步进行多边形逼近,进而获得一个凸多边形。找到重心位置,通过比较凸包的顶点与重心的y轴坐标,得到纵坐标大于重心的凸包的顶点,再规定凸点间距离范围,得到指尖数量。

convexHull()函数原型
convexHull(  InputArray      points,
OutputArray hull,
bool            clockwise = false,
bool            returnPoints = true
)

①points:输入的二维点集或轮廓坐标,数据类型为vector或者Mat。
②hull:输出的凸包的顶点的坐标或者索引,数据类型为vector或者vector。
③clockwise:方向标志。当参数取值为true时,凸包顺序为顺时针方向;当参数取值为false时,凸包顺序为逆时针方向。
④returnPoints:输出数据的类型标志。当参数取值为true时,第二个参数输出的结果是凸包顶点的坐标,数据类型为vector;当参数取值为false时,第二个参数输出的结果是凸包顶点的索引,数据类型为vector。

5.2moments()函数原型

moments()函数原型
moments(    InputArray      array,
bool            binaryImage = false
)

①array:计算矩的区域二维像素坐标集合或者单通道的CV_8U图像。
②binaryImage:是否将所有非零像素值视为1的标志,该标志只在第一个参数设置为图像类型的数据时才会起作用。
moments()函数会返回一个Moments类的变量。Moments类中含有几何矩、中心矩及归一化的几何矩的数值属性。

5.3Mouse_event()函数原型

本实验通过mouse_event()函数来代替鼠标操作。

Mouse_event()函数原型
mouse_event(    DWORD       dwFlags,
DWORD       dx,
DWORD       dy,
DWORD       dwData,
ULONG_PTR   dwExtraInfo
)

①dwFlags:标志位集,指定点集按钮和鼠标动作。
②dx:指定鼠标沿x轴的绝对位置或者从上次鼠标事件产生以来移动的数量。
③dy:指定鼠标沿y轴的绝对位置或者从上次鼠标事件产生以来移动的数量。
④dwData:如果dwFlags为MOUSEEVENT_WHEEL,则dwData指定鼠标轮移动的数量。正值表示鼠标轮向前移动,即远离用户的方向;负值表示鼠标轮向后移动,即朝向用户。如果dwFlags不是MOUSEEVENT_WHEEL,则dwData应为0。
⑤dwExtrafo:指定与鼠标事件相关的附加32位值。

鼠标动作标志参数 含义
MOUSEEVENT_ABSOLUTE dx和dy参数含有规范化的绝对坐标。如果不设置,这些参数含有相对数据:相对于上次位置的改动位置
MOUSEEVENT_MOVE 鼠标移动
MOUSEEVENT_LEFTDOWN 鼠标左键按下
MOUSEEVENT_LEFTUP 鼠标左键松开
MOUSEEVENT_RIGHTDOWN 鼠标右键按下
MOUSEEVENT_RIGHTUP 鼠标右键松开
MOUSEEVENT_MIDLEDOWN 鼠标中键按下
MOUSEEVENT_MIDLEUP 鼠标中键松开
MOUSEEVENT_WHEEL 鼠标轮被滚动,如果鼠标有一个滚轮,滚轮数量由dwData给出

5.5 Code

/*凸包检测+重心+ 鼠标操作 */
Mat CH(Mat&Image_src)
{/*轮廓*/Mat ImageOut = approx(Image_src);vector<vector<Point>>contours;vector<Vec4i>hierarchy;findContours(ImageOut, contours, hierarchy, 0, 2, Point());/*画重心*/Moments moment = moments(ImageOut, true);Point center(moment.m10 / moment.m00, moment.m01 / moment.m00);circle(ImageOut, center, 8, Scalar(255, 255, 255), -1);int dist;int sum = 0;for (int t = 0;t < contours.size();t++){/*凸包检测*/vector<Point>hull;convexHull(contours[t], hull);for (size_t i = 0;i < hull.size();i++){int a = hull.size();if (i != hull.size() - 1)dist = distance(hull[i], hull[i + 1]);int dist1 = distance(hull[i], center);if (hull[i].y < center.y&&dist>20){circle(ImageOut, hull[i], 15, Scalar(255, 255, 255), 2, 8, 0);sum += 1;}if (i == hull.size() - 1){line(ImageOut, hull[i], hull[0], Scalar(255, 255, 255), 5, 8, 0);break;}line(ImageOut, hull[i], hull[i + 1], Scalar(255, 255, 255), 5, 8, 0);}}cout << sum << endl;if (sum == 1){mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);string str3 = "Left";putText(ImageOut, str3, center, 0, 2, Scalar(255, 255, 255), 4, 8);waitKey(0);}if (sum == 2){mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);mouse_event(MOUSEEVENTF_LEFTDOWN | MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);string str1 = "Double click";putText(ImageOut, str1, center, 0, 2, Scalar(255, 255, 255), 4, 8);}if (sum == 3){mouse_event(MOUSEEVENTF_RIGHTDOWN | MOUSEEVENTF_RIGHTDOWN, 0, 0, 0, 0);string str2 = "Right";putText(ImageOut, str2, center, 0, 2, Scalar(255, 255, 255), 4, 8);waitKey(0);}return ImageOut;
}

6.主函数 摄像头调用

6.1摄像头调用

OpenCV中为读取视频文件和调用摄像头而设计了VideoCapture类。视频文件由专门的视频读取函数进行视频读取,并将每一帧图像保存到Mat类矩阵中。
本实验通过VideoCapture类直接调用笔记本内置摄像头。

VideoCapture类调用摄像头构造函数
VideoCapture( int       index,
Int     apiPreference
)

①index:需要打开的摄像头设备的ID。
②airPreference:读取数据时设置的属性。

6.2 Code

int main()
{Introduce();/*调用摄像头*/VideoCapture capture(0);/*检查是否成功打开*/if (!capture.isOpened()){cout << "摄像头打开失败T_T";return -1;}while (1) {Mat In;capture >> In;Mat A = skin(In);Mat B = Floodfill(A);Mat Out = CH(B);namedWindow("Result", WINDOW_NORMAL | WINDOW_KEEPRATIO);imshow("Result", Out);waitKey(2);}return 0;
}

7.代码中的其他API函数

7.1 getStructuringElement()函数原型

getStructuringElement(   int         shape,
Size        ksize,
Point       anchor = Point(-1,-1)
)

①shape:生成结构元素的种类,可选参数及其含义如表所示。
②ksize:结构元素的尺寸。
③anchor:中心点的位置,默认为结构元素的几何中心。
标志参数 简记 作用
MORPH_RECT 0 矩形结构元素,所有元素都为1
MORPH_CROSS 1 十字结构元素,中间的列和行元素为1
MORPH_ELLIPSE 2 椭圆结构元素,矩形的内接椭圆元素为1

7.2 findContours()函数原型

FindCountours(    InputArray                  image,
OutputArrayOfArrays     contours,
OutputArray             hierarchy,
int                         mode,
int                         method,
Point                       offset = Point()
)

①image:输入图像,数据类型为CV_8U的单通道灰度图或二值化图像。
②contours:存放检测到的轮廓,每个轮廓中放着像素的坐标。数据类型为vector<vector>。
③hierarchy:存放各个轮廓之间的结构信息,数据类型为vector。
④mode:轮廓检测模式标志。
⑤method:轮廓逼近方法标志。
⑥offset:每个轮廓点移动的可选偏移量。

轮廓检测模式标志参数 简记 含义
RETR_EXTERNAL 0 只检测最外层轮廓
RETR_LIST 1 提取所有轮廓,并放在list中。检测的轮廓不建立等级关系
RETR_CCOMP 2 提取所有轮廓,并且将其组织为双层结构。顶层为连通域的外围边界,次层为孔的内层边界
RETR_TREE 3 提取所有轮廓,并重新建立网状的轮廓结构
轮廓逼近方法标志参数 简记 含义
CHAIN_APPROX_NONE 1 获取每个轮廓的每个像素,相邻两个点的像素位置相差1
CHAIN_APPROX_SIMPLE 2 压缩水平方向、垂直方向和对角线方向的元素,只保留该方向的终点坐标

7.3 circle()函数原型

circle(   InputOutputArray    img,
Point                   center,
int                     radius,
const                   Scalar &color,
int                     thickness = 1,
int                     lineType = LINE_8,
int                     shift = 0
)

①img:需要绘制圆形的图像。
②center:圆形的圆心位置坐标。
③radius:圆形的半径,单位为像素。
④color:圆形的颜色。
⑤thickness:轮廓的宽度,如果数值为负,则绘制一个实心圆。
⑥lineType:边界类型。
⑦shift:中心坐标和半径数值中的小数位数。

7.4 line()函数原型

line( InputOutputArray        img,
Point                   pt1,
Point                   pt2,
const                   scalar & color,
int                     thickness = 1,
int                     lineType = LINE_8,
int                     shift = 0
)

①pt1:直线起点在图像中的坐标。
②pt2:直线终点在图像中的坐标。
③color:直线的颜色。

7.5 namedWindow()函数

namedWindow(  const   String & winname,
int     flags = WINDOW_AUTOSIZE
)

①winname:窗口名称,用作窗口的标识符。
②flags:窗口属性设置标志。在默认的情况之下,窗口所加载的标志参数为“WINDOW_AUTOSIZE|WINDOW_KEEPRATIO|WINDOW_GUI_EXPANDED”。

窗口属性标志参数 作用
WINDOW_NORMAL 显示图像后,允许用户随意调整窗口大小
WINDOW_AUTOSIZE 根据图像大小显示窗口,不允许用户调整大小
WINDOW_OPENGL 创建窗口的时候会支持OpenGL
WINDOW_FULLSCREEN 全屏显示窗口
WINDOW_FREERATIO 调整图像尺寸以充满窗口
WINDOW_KEEPRATIO 保持图像的比例
WINDOW_GUI_EXPANDED 创建的窗口允许添加工具栏和状态栏
WINDOW_GUI_NORMAL 创建没有状态栏和工具栏的窗口

7.6 imshow()函数

imshow(   const       String & winname,
InputArray  mat
)

①winnam:要显示图像的窗口的名字,用字符串形式赋值。
②mat:要显示的图像矩阵。

8.参考文献

[1]贾建军.基于视觉的手势识別技术研究[D].哈尔滨工业大学.2008
[2]孟国庆.基于OpenCV的手势识别技术研究[D].西安科技大学.2014
[3]Gary Bradski Adrian Kaehler .学习OpenCV[M]. 于仕琪 刘瑞祯译. 北京:清华大学出版社.2014
[4]王天庆. Python人脸识别从入门到工程实践[M]. 北京:机械工业出版社.2019.4
[5]冯振 郭延宁 吕跃勇. OpenCV 4 快速入门[M].北京:人民邮电出版社.2020

基于OpenCV的简易实时手势识别(含代码)相关推荐

  1. python实现流媒体传输_基于OpenCV的网络实时视频流传输的实现

    很多小伙伴都不会在家里或者办公室安装网络摄像头或监视摄像头.但是有时,大家又希望能够随时随地观看视频直播. 大多数人会选择使用IP摄像机(Internet协议摄像机)而不是CCTV(闭路电视),因为它 ...

  2. android opencv 获取小图在大图的坐标_Android开发—基于OpenCV实现相机实时图像识别跟踪...

    利用OpenCV实现实时图像识别和图像跟踪 图像识别 什么是图像识别 图像识别,是指利用计算机对图像进行处理.分析和理解,以识别各种不同模式的目标和对像的技术.根据观测到的图像,对其中的物体分辨其类别 ...

  3. 【opencv】基于opencv实现人脸识别,原理代码部分

    上一部分我们解决了环境问题,这一部分我们可以开始上代码,环境没有配好的可以参照上一篇博客:环境搭建解决: 下面先说一下原理: 一.原理部分   本文基于opencv来实现人脸识别,大致实现流程可以描述 ...

  4. 基于Opencv的简易手势识别

    这是我作为期末的python大作业设计的作业,在此感谢CSDN上的大佬,网上类似的程序有不少,学习借鉴后经过自己的整理,程序比较简陋,没有大佬们做的那么丰富,属于个人应付本科课程设计的作业,这篇文章也 ...

  5. 基于 OpenCV 的网络实时视频流传输

    作者 | 努比 来源 | 小白学视觉 大多数人会选择使用IP摄像机(Internet协议摄像机)而不是CCTV(闭路电视),因为它们具有更高的分辨率并降低了布线成本.在本文中,我们将重点介绍IP摄像机 ...

  6. 基于OpenCV的网络实时视频流传输

    点击上方"小白学视觉",选择加"星标"或"置顶" 重磅干货,第一时间送达 很多小伙伴都不会在家里或者办公室安装网络摄像头或监视摄像头.但是有 ...

  7. 基于opencv和mfc的摄像头采集代码(GOMFCTemplate2)持续更新

            编写带界面的图像处理程序,选择opencv+mfc是一种很好的选择:在读取摄像头数据方面,网上的方法很多,其中shiqiyu的camerads的方法是较好的.       基于现有资料 ...

  8. 基于opencv的车牌识别解析与代码

    Jeremy Lin @HQU 车牌识别太出名了,我也就花几天来了解下这个系统,并结合opencv进行实现.下面是一些介绍: 车辆牌照识别(License Plate Recognition,LPR) ...

  9. 基于opencv和QT的摄像头采集代码( GoQTtemplate3持续更新)

    在Linux操作系统上,编写带界面的图像处理程序,选择opencv+QT是一种很好的选择.GoQTtemplate3是我为编写Linux下图像处理程序实现的框架,希望能够为大家解决Linux环境下桌面 ...

最新文章

  1. Android RatingBar
  2. 异步请求之XMLHttpRequest篇
  3. HTTP报文简单介绍
  4. Python入门学习笔记13(线程锁与信号量)
  5. boost::mp11::mp_reverse相关用法的测试程序
  6. ubuntu16.04 关闭防火墙的方法
  7. 单片机点灯不难,但你会网页点灯吗?
  8. 依赖: ros-melodic-desktop 但是它将不会被安装_npm系列之依赖管理
  9. MongoDB无法注册windows服务问题解决 Error connecting to the Service Control Manager: 拒绝访问 Mongodb M
  10. NXOpen.BlockStyler的一些用法
  11. maven项目包导不进去_IntelliJ Idea下Maven插件使用技巧
  12. Html select、option、optgroup 标签
  13. elementUI 学习入门之 Button 按钮
  14. 国开大学计算机原理边学边练,国家开放大学计算机组成原理
  15. m序列的产生原理及其性质
  16. 微软所有正版产品下载地址
  17. 彻底清除计算机病毒的方式是,彻底清除电脑病毒的小方法
  18. 运维小知识---If you insist running as root, then set the environment variable RUN_AS_USER=root......
  19. 周礼键君:中国福建省建瓯市之《闽郡八音字典》修订版
  20. Python 自动化办公-玩转 Word

热门文章

  1. 中国电信移动物联网发展成果与创新实践 ,干货满满
  2. 一生的读书计划——影响中国历史进程的中国名人2
  3. 弹性ip是可以加入共享带宽?
  4. 结构化软件设计的具体步骤
  5. 罗振宇2021《时间的朋友》体会
  6. Python学习:给类或者类的对象添加打印内容 def __repr__(self)
  7. [基于富瀚6630平台实现U盘热插拔检测并mount]
  8. linux内核的红黑树
  9. 第一章软件危机与软件工程
  10. 留学人员入户佛山需要哪些材料