开始

cv版本2.4.9

编译器vs2019

资料书:OpenCV3编程入门(毛星云)

配置

按照网上教程。

遇到的问题

运行测试程序遇到的问题:

OpenCV Error: Assertion failed (size.width>0 && size.height>0) in cv::imshow, file …\opencv\modules\highgui\src\window.cpp, line 261

原因是由于图片的地址错误。图片应该存在项目下。

测试程序

#include <opencv2/opencv.hpp>
using namespace cv;
int main() {/*Mat img = imread("1.jpg");*///这里的图片存在项目下(文件中)Mat img = imread("D:\\cv\\opencv_demo\\1.jpg");//注意地址间是\\imshow("[载入的图片]", img);waitKey(6000);
}

在运行过程中,如果出现XXX未定义等情况,最好将cv删掉重新解压使用

简单几个例子

1、图像腐蚀

用暗色部分腐蚀掉图片中的亮色部分

#include <opencv2/opencv.hpp>
using namespace cv;
//图片腐蚀
int main() {Mat scrimage = imread("1.jpg");imshow("原图", scrimage);//腐蚀操作Mat element = getStructuringElement(MORPH_RECT, Size(15, 15));Mat dstImage;erode(scrimage, dstImage, element);//进行图像腐蚀imshow("效果图", dstImage);waitKey();return 0;
}

2、图像模糊

#include <opencv2/opencv.hpp>
using namespace cv;
//图片模糊
int main() {Mat scrImage = imread("1.jpg");imshow("原图", scrImage);Mat dstImage;blur(scrImage, dstImage, Size(7, 7));//图片模糊imshow("效果图", dstImage);waitKey();
}

3、canny边缘检测

载入图像,先转为灰度图,再用blur进行图片模糊降噪,用canny进行边缘检测

#include <opencv2/opencv.hpp>
using namespace cv;
//图片边缘检测
int main() {Mat srcImage = imread("2.jpg");imshow("原图", srcImage);Mat dstImage,edge,grayImage;dstImage.create(srcImage.size(), srcImage.type());//设置和原图一样的矩阵cvtColor(srcImage, grayImage, CV_BGR2GRAY);//在dstImage矩阵上生成图blur(grayImage, edge, Size(3, 3));Canny(edge, edge, 3, 9, 3);imshow("效果图", edge);waitKey();return 0;
}

4、读取并播放视频

#include <opencv2\opencv.hpp>
using namespace cv;
int main(){VideoCapture capture("1.avi");//存储位置相同while(1){Mat frame;capture>>frame;imshow("读取视频",frame);waitKey(30);}return 0;
}

5、截取摄像头图像并canny处理

#include <opencv2/opencv.hpp>
using namespace cv;
int main() {VideoCapture capture(0);//表示使用摄像头Mat edges;while (1) {Mat frame;capture >> frame;cvtColor(frame, edges, CV_BGR2GRAY);blur(edges, edges, Size(7, 7));Canny(edges, edges, 0, 30, 3);imshow("结果", edges);if (waitKey(30) >= 0) {break;}}return 0;
}
//输出当前cv版本
printf("\t版本 OpenCV" CV_VERSION);

HighGUI图形用户界面初步

cv2开始使用Mat为数据类型进行图像存取

图像的载入、显示及输出

imread()

读取文件中的图像

Mat imread(const string& filename,int flags=1);

const string&filename:填入图片的路径名

int flags:载入标识,指定加载图像的颜色类型。默认值为1,可忽略。

取个各值意义:

​ -1:废除

​ 0:图像转换成灰度后返回

​ 1:转换图像到彩色再返回(默认值)

​ 2:若取该值且载入的图像的深度为16或32位,就返回对应深度的图像,否则就转换为8位图像再返回

如果不想使用以上固定的赋值方式,还可:

​ falgs>0 返回三通道的彩色图像

​ flags=0 返回灰度图像(就是灰色的)

​ flags<0返回包含Alpha通道的加载图像

imshow()

在指定窗口中显示一幅图像

void imshow(const string& winname,InputArray mat)

const strings& winname:需要显示的窗口的标识名称

InputArray mat:需要显示的图像

如果窗口是用CV_WINDOW_AUTOSIZE(默认值)标志创建,那么显示图像的原始大小,否则将图像进行缩放以适合窗口。

注意:在这里遇到的InputArray类型可以直接看作是Mat类型来使用

namedWindow()

创建一个窗口

如果只是简单的显示图像,不需要这一步,直接imread、imshow即可。但是如果需要在显示窗口之前就用到窗口名时,如添加滑动条,就需要先创建出窗口,显式规定窗口名称

void namedWindow(const string& winname,int flags = WINDOW_AUTOSIZE)

winname:填写被用作窗口的标识符的窗口名称

flags:窗口的标识,它可以填以下值

​ WINDOW_NORMAL:用户可以改变窗口的大小(或者是CV_WINDOW_NORMAL)

​ WINDOW_AUTOSIZE:自动调整窗口大小,用户不可手动修改(默认

​ WINDOW_OPENGL: 窗口创建时会支持Open GUI

以上标识,在CV2中也可写为CV_标识

注意namedwindow建的窗口要与后续imread显示图片使用的名字一样

imwrite()

输出图像到文件

bool imwrite(const string& filename,InputArray img,const vector& params=vector())

filename:写入的文件名。要带上后缀

img:Mat数据类型的图像数据

params:特定格式保存的参数编码。有默认值。要填写的话查一下。

#include <opencv2/opencv.hpp>
#include <vector>
using namespace cv;
using namespace std;
void createAlphaMat(Mat& mat)
{for (int i = 0; i < mat.rows; ++i) {for (int j = 0; j < mat.cols; ++j) {Vec4b& rgba = mat.at<Vec4b>(i, j);rgba[0] = UCHAR_MAX;rgba[1] = saturate_cast<uchar>((float(mat.cols - j)) / ((float)mat.cols) * UCHAR_MAX);rgba[2] = saturate_cast<uchar>((float(mat.rows - i)) / ((float)mat.rows) * UCHAR_MAX);rgba[3] = saturate_cast<uchar>(0.5 * (rgba[1] + rgba[2]));}}
}
int main()
{//创建带alpha通道的MatMat mat(480, 640, CV_8UC4);createAlphaMat(mat);vector<int>compression_params;compression_params.push_back(CV_IMWRITE_PNG_COMPRESSION);//CV2版本//compression_params.push_back(IMWRITE_PNG_COMPRESSION);//CV3版本compression_params.push_back(9);
//这里是对imwrite中params参数的输入设置//显示图片try {imwrite("透明Alpha值图.png", mat, compression_params);imshow("生成的png图", mat);fprintf(stdout, "PNG图片文件的alpha数据保存完毕~\n可以在工程目录下查看由imwrite函数生成的图片\n");waitKey(0);}catch (runtime_error& ex) {fprintf(stderr, "图像转换成PNG格式发生错误:%s\n", ex.what());return 1;}return 0;
}

上述代码重点在于imwrite的使用方式,不需要纠结它的画法

CV_IMWRITE_PNG_COMPRESSION:png格式图片

CV_IMWRITE_JPEG_QUALITY:jpeg格式

图像混合

#include <opencv2/opencv.hpp>
#include <vector>
using namespace cv;
using namespace std;
int main()
{Mat image = imread("dota.jpg", 199);Mat logo = imread("1.jpg");namedWindow("原图");imshow("原图", image);Mat imageROI;//imageROI = image(Rect(800, 350, logo.cols, logo.rows));imageROI = image(Range(100, 100 + logo.rows), Range(200, 200 + logo.cols));//100和200表示起始点的位置addWeighted(imageROI, 0.5, logo, 0.3, 0., imageROI);namedWindow("修改后");imshow("修改后", image);imwrite("生成.jpg", image);waitKey(0);return 0;
}

将两张图片贴在一起。需要注意的点是使用Range计算logo图所贴在dota图上位置时,要计算好位置,否则会出现越界的错误

滑动条创建

依附于窗口存在

createTrackbar()

创建一个可以调整数值的滑动条

int createTrackbar(const string& trackbarname,const string& winname,int value,int count,TrackbarCallback onChange=0,void userdata=0)**

trackbarname:轨迹条的名字

winname:窗口的名字(对应namedWindow)

value:指向整型的指针,表示滑块位置

count:表示滑块可以达到的最大位置的值。滑块最小位置的值为0.

onChange:默认为0.指向回调函数指针,滑块位置改变函数就回调。函数回调原型必须是

*void XX(int,void)**形式

userdata:默认为0.用户传回回调函数的数据,用来处理轨迹条事件。如果value为全局变量,则不需要管该函数

#include <opencv2/opencv.hpp>
#include <vector>
using namespace cv;
using namespace std;
//滑动条控制混合图像
#define WINDOW_NAME "线性混合示例"
//这里定义的窗口名为了在函数和main中都可以指向同一个窗口
const int g_nMaxAlphaValue = 100;//Alpha值的最大值
int g_nAlphaValueSlider;//滑动条对应的变量
double g_dAlphaValue;
double g_dBetaValue;//声明存储图像的变量
Mat g_srcImage1;
Mat g_srcImage2;
Mat g_dstImage;void on_Trackbar(int, void*)
{//求出当前alpha值相对于最大值的比例g_dAlphaValue = (double)g_nAlphaValueSlider / g_nMaxAlphaValue;//则beta值为1减去alpha值g_dBetaValue = (1.0 - g_dAlphaValue);//根据alpha和beta值进行线性混合addWeighted(g_srcImage1, g_dAlphaValue, g_srcImage2, g_dBetaValue, 0.0, g_dstImage);//显示效果图imshow(WINDOW_NAME, g_dstImage);
}
int main(int argc, char** argv)
{//加载图像 (两图像的尺寸需相同)g_srcImage1 = imread("3.1.jpg");g_srcImage2 = imread("3.2.jpg");if (!g_srcImage1.data) { printf("读取第一幅图片错误,请确定目录下是否有imread函数指定图片存在~! \n"); return -1; }if (!g_srcImage2.data) { printf("读取第二幅图片错误,请确定目录下是否有imread函数指定图片存在~!\n"); return -1; }//设置滑动条初值为70g_nAlphaValueSlider = 70;//创建窗体namedWindow(WINDOW_NAME, 1);//在创建的窗体中创建一个滑动条控件char TrackbarName[50];//滑动条前说明字符的长度sprintf(TrackbarName, "透明值 %d", g_nMaxAlphaValue);createTrackbar(TrackbarName, WINDOW_NAME, &g_nAlphaValueSlider, g_nMaxAlphaValue, on_Trackbar);//这里的第三个参数value为全局变量,所以最后一个参数可以省略。on_Trackbar为回调函数//结果在回调函数中显示on_Trackbar(g_nAlphaValueSlider, 0);///显示的拖动的值。刚打开为默认值//按任意键退出waitKey(0);return 0;
}

可以通过上面的滑动条修改图片的透明度

getTrackbarPos()

获取当前轨迹条的位置并返回。

int getTrackbarPos(const string& trackname,const string& winname):trackname为轨迹条名字,winname为轨迹条父窗口名字

鼠标操作

void setMouseCallback(const string& winname,MouseCallback onMouse,void* userdata = 0)

winname:窗口名字

onmouse:指定窗口里每次鼠标时间发生的时候,被调用的函数指针

userdata:用户定义的传递到回调函数的参数,默认值为0

#include <opencv2/opencv.hpp>
using namespace cv;#define WINDOW_NAME "【程序窗口】"        //为窗口标题定义的宏 //-----------------------------------【全局函数声明部分】------------------------------------
//      描述:全局函数的声明
//------------------------------------------------------------------------------------------------
void on_MouseHandle(int event, int x, int y, int flags, void* param);
void DrawRectangle(cv::Mat& img, cv::Rect box);//-----------------------------------【全局变量声明部分】-----------------------------------
//      描述:全局变量的声明
//-----------------------------------------------------------------------------------------------
Rect g_rectangle;
bool g_bDrawingBox = false;//是否进行绘制
RNG g_rng(12345);//-----------------------------------【main( )函数】--------------------------------------------
//      描述:控制台应用程序的入口函数,我们的程序从这里开始执行
//-------------------------------------------------------------------------------------------------
int main(int argc, char** argv)
{//【1】准备参数g_rectangle = Rect(-1, -1, 0, 0);Mat srcImage(600, 800, CV_8UC3), tempImage;srcImage.copyTo(tempImage);g_rectangle = Rect(-1, -1, 0, 0);srcImage = Scalar::all(0);//【2】设置鼠标操作回调函数namedWindow(WINDOW_NAME);setMouseCallback(WINDOW_NAME, on_MouseHandle, (void*)&srcImage);//【3】程序主循环,当进行绘制的标识符为真时,进行绘制while (1){srcImage.copyTo(tempImage);//拷贝源图到临时变量if (g_bDrawingBox) DrawRectangle(tempImage, g_rectangle);//当进行绘制的标识符为真,则进行绘制imshow(WINDOW_NAME, tempImage);if (waitKey(10) == 27) break;//按下ESC键,程序退出}return 0;
}//--------------------------------【on_MouseHandle( )函数】-----------------------------
//      描述:鼠标回调函数,根据不同的鼠标事件进行不同的操作
//-----------------------------------------------------------------------------------------------
void on_MouseHandle(int event, int x, int y, int flags, void* param)
{Mat& image = *(cv::Mat*) param;switch (event){//鼠标移动消息case EVENT_MOUSEMOVE:{if (g_bDrawingBox)//如果是否进行绘制的标识符为真,则记录下长和宽到RECT型变量中{g_rectangle.width = x - g_rectangle.x;g_rectangle.height = y - g_rectangle.y;}}break;//左键按下消息case EVENT_LBUTTONDOWN:{g_bDrawingBox = true;g_rectangle = Rect(x, y, 0, 0);//记录起始点}break;//左键抬起消息case EVENT_LBUTTONUP:{g_bDrawingBox = false;//置标识符为false//对宽和高小于0的处理if (g_rectangle.width < 0){g_rectangle.x += g_rectangle.width;g_rectangle.width *= -1;}if (g_rectangle.height < 0){g_rectangle.y += g_rectangle.height;g_rectangle.height *= -1;}//调用函数进行绘制DrawRectangle(image, g_rectangle);}break;}
}
//-----------------------------------【DrawRectangle( )函数】------------------------------
//      描述:自定义的矩形绘制函数
//-----------------------------------------------------------------------------------------------
void DrawRectangle(cv::Mat& img, cv::Rect box)
{cv::rectangle(img, box.tl(), box.br(), cv::Scalar(g_rng.uniform(0, 255), g_rng.uniform(0, 255), g_rng.uniform(0, 255)));//随机颜色
}

数据结构及基本绘图

图片在设备中是以像素点矩阵的形式存在的

Mat

Mat是一个类,有两个数据部分组成:矩阵头和一个指向存储所有像素值的矩阵的指针。

自动开辟空间,不需要手动释放空间

在使用图像处理函数时,会在函数中传递大量的图片。传递大量图片时图片的复制会降低程序的运行速度。所以引入了引用计数机制。

引用计数机制:每个Mat对象有自己的信息头,但是共享同一个矩阵。这通过让矩阵指针指向同一地址而实现。拷贝构造函数只复制信息头和矩阵指针,而不复制矩阵。

Mat A,C;
A = imread("1.jpg");
c = A;//这里C和A指向的都是同一个矩阵,任意一个做出修改都会导致矩阵发生变化

当矩阵属于多个对象时,由最后一个对象来清理它

如果想要复制矩阵本身,则可以使用函数clone()或者copyTo()

Mat F = A.clone();
Mat C;
A.copyTo(C);//A为被复制的矩阵

++++++++++++++++++++++++++++++++++++++++++++

存储像素值需要指定颜色空间和数据类型。

颜色系统有很多:

  1. RGB:最常见
  2. HSV和HLS:把颜色分为色调、饱和度和亮度、明度。对光照条件不敏感
  3. YCrCb:在JPEG格式中常用

如何存储一个原色决定在其定义域上能够控制的精度。最小为char,也可以用有符号型或无符号型。RGB中 三个char已经可以表示1600万种可能的颜色。但是如果用float或double又能给出更加精密的颜色分辨能力。

增加颜色尺寸也会增加图像所占的内存空间

显式创建对象

Mat不仅是图像容器类,同时也是一个通用的矩阵类。

  • 使用Mat()构造函数

Mat(行数,列数,CV_[][][位数] [带符号与否] [类型前缀][通道数])

Mat M(2,2,CV_8UC3,Scalar(0,0,255));
cout<<M<<endl;
//运算符<<只对二维矩阵有效
//Scalar为short型的向量,指定初始值//构造多维
int sz[3] = {2,2,2};
Mat L(3,sz,CV_8UC,Scalar::all(0));//使用指针创建信息头
IPlImage* img = cvLoadImage("1.jpg",1);
Mat mtx(img);//create函数
M.create(4,4,CV_8UC(2));//此方法不能为矩阵设初值,只是改变尺寸时重新为矩阵数据开辟内存Mat F = Mat::eye(4,4,CV_64F);Mat C = (Mat_<double>(3,3) << 0,-1,0,-1,5,-1,2,2,2);Mat RowClone = C.row(1).clone();

常用数据结构

Point类

表示二维坐标系下的点,即以x,y来表示

Point point;
point.x = 1;
point.y = 2;
//或者
Point point = Point(1,2);

Scalar类

用于传递像素值,它可以表示有4个像素的值,但如果只写三个也可以。

如Scalar(a,b,c)

在RGB中 a为蓝色分量,b绿色,c为红色

Size类

表示尺寸

Rect类

表示矩阵。

其成员变量有x、y、width、height表示左上角的坐标和矩形的宽和高

Size返回Size;area()返回面积;contains(Point)看点是否在矩形内;inside(Rect函数判断矩形是否在矩形内);

tl()返回左上角坐标;br返回右下角坐标

颜色空间的转换

使用cvtColor()函数

cvtColor(输入图像,输出图像,颜色空间转换标识符,目标对象的通道数(若为0则表示目标图像取源图像的通道数))

Opencv中默认的图片通道存储顺序为BGR

#include<iostream>
#include<opencv2/opencv.hpp>using namespace cv;
using namespace std;int main()
{Mat image1 = imread("1.1.jpg"),dstImage;  //读取图像;if (image1.empty()){cout << "读取错误" << endl;return -1;}cvtColor(image1, dstImage, COLOR_BGR2Lab);imshow("xiaog", dstImage);waitKey(0);return 0;
}

基本图形绘制

在绘制前,程序源文件开头会有:

#define WINDOW_WIDTH 600 //定义窗口大小的宏

#include <opencv2/core/core.hpp>
#include <opencv2/highgui/highgui.hpp>
using namespace cv;//OpenCV3需加入头文件:
#include <opencv2/imgproc/imgproc.hpp>
//-----------------------------------【宏定义部分】--------------------------------------------
//      描述:定义一些辅助宏
//------------------------------------------------------------------------------------------------
#define WINDOW_NAME1 "【绘制图1】"        //为窗口标题定义的宏
#define WINDOW_NAME2 "【绘制图2】"        //为窗口标题定义的宏
#define WINDOW_WIDTH 600//定义窗口大小的宏void DrawEllipse(Mat img, double angle);//绘制椭圆
void DrawFilledCircle(Mat img, Point center);//绘制圆
void DrawPolygon(Mat img);//绘制多边形
void DrawLine(Mat img, Point start, Point end);//绘制线段int main(void)
{// 创建空白的Mat图像Mat atomImage = Mat::zeros(WINDOW_WIDTH, WINDOW_WIDTH, CV_8UC3);Mat rookImage = Mat::zeros(WINDOW_WIDTH, WINDOW_WIDTH, CV_8UC3);// ---------------------<1>绘制化学中的原子示例图------------------------//【1.1】先绘制出椭圆DrawEllipse(atomImage, 90);//画椭圆,90表示该椭圆的旋转读数DrawEllipse(atomImage, 0);DrawEllipse(atomImage, 45);DrawEllipse(atomImage, -45);//【1.2】再绘制圆心DrawFilledCircle(atomImage, Point(WINDOW_WIDTH / 2, WINDOW_WIDTH / 2));// ----------------------------<2>绘制组合图-----------------------------//【2.1】先绘制出椭圆DrawPolygon(rookImage);// 【2.2】绘制矩形rectangle(rookImage,Point(0, 7 * WINDOW_WIDTH / 8),Point(WINDOW_WIDTH, WINDOW_WIDTH),Scalar(0, 255, 255),-1,8);// 【2.3】绘制一些线段DrawLine(rookImage, Point(0, 15 * WINDOW_WIDTH / 16), Point(WINDOW_WIDTH, 15 * WINDOW_WIDTH / 16));DrawLine(rookImage, Point(WINDOW_WIDTH / 4, 7 * WINDOW_WIDTH / 8), Point(WINDOW_WIDTH / 4, WINDOW_WIDTH));DrawLine(rookImage, Point(WINDOW_WIDTH / 2, 7 * WINDOW_WIDTH / 8), Point(WINDOW_WIDTH / 2, WINDOW_WIDTH));DrawLine(rookImage, Point(3 * WINDOW_WIDTH / 4, 7 * WINDOW_WIDTH / 8), Point(3 * WINDOW_WIDTH / 4, WINDOW_WIDTH));// ---------------------------<3>显示绘制出的图像------------------------imshow(WINDOW_NAME1, atomImage);moveWindow(WINDOW_NAME1, 0, 200);imshow(WINDOW_NAME2, rookImage);moveWindow(WINDOW_NAME2, WINDOW_WIDTH, 200);waitKey(0);return(0);
}//-------------------------------【DrawEllipse( )函数】--------------------------------
//      描述:自定义的绘制函数,实现了绘制不同角度、相同尺寸的椭圆
//-----------------------------------------------------------------------------------------
void DrawEllipse(Mat img, double angle)
{int thickness = 2;int lineType = 8;ellipse(img,Point(WINDOW_WIDTH / 2, WINDOW_WIDTH / 2),Size(WINDOW_WIDTH / 4, WINDOW_WIDTH / 16),angle,0,360,Scalar(255, 129, 0),thickness,lineType);
}//-----------------------------------【DrawFilledCircle( )函数】---------------------------
//      描述:自定义的绘制函数,实现了实心圆的绘制
//-----------------------------------------------------------------------------------------
void DrawFilledCircle(Mat img, Point center)
{int thickness = -1;int lineType = 8;circle(img,center,WINDOW_WIDTH / 32,Scalar(0, 0, 255),thickness,lineType);
}//-----------------------------------【DrawPolygon( )函数】--------------------------
//      描述:自定义的绘制函数,实现了凹多边形的绘制
//--------------------------------------------------------------------------------------
void DrawPolygon(Mat img)
{int lineType = 8;//创建一些点Point rookPoints[1][20];rookPoints[0][0] = Point(WINDOW_WIDTH / 4, 7 * WINDOW_WIDTH / 8);rookPoints[0][1] = Point(3 * WINDOW_WIDTH / 4, 7 * WINDOW_WIDTH / 8);rookPoints[0][2] = Point(3 * WINDOW_WIDTH / 4, 13 * WINDOW_WIDTH / 16);rookPoints[0][3] = Point(11 * WINDOW_WIDTH / 16, 13 * WINDOW_WIDTH / 16);rookPoints[0][4] = Point(19 * WINDOW_WIDTH / 32, 3 * WINDOW_WIDTH / 8);rookPoints[0][5] = Point(3 * WINDOW_WIDTH / 4, 3 * WINDOW_WIDTH / 8);rookPoints[0][6] = Point(3 * WINDOW_WIDTH / 4, WINDOW_WIDTH / 8);rookPoints[0][7] = Point(26 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 8);rookPoints[0][8] = Point(26 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 4);rookPoints[0][9] = Point(22 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 4);rookPoints[0][10] = Point(22 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 8);rookPoints[0][11] = Point(18 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 8);rookPoints[0][12] = Point(18 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 4);rookPoints[0][13] = Point(14 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 4);rookPoints[0][14] = Point(14 * WINDOW_WIDTH / 40, WINDOW_WIDTH / 8);rookPoints[0][15] = Point(WINDOW_WIDTH / 4, WINDOW_WIDTH / 8);rookPoints[0][16] = Point(WINDOW_WIDTH / 4, 3 * WINDOW_WIDTH / 8);rookPoints[0][17] = Point(13 * WINDOW_WIDTH / 32, 3 * WINDOW_WIDTH / 8);rookPoints[0][18] = Point(5 * WINDOW_WIDTH / 16, 13 * WINDOW_WIDTH / 16);rookPoints[0][19] = Point(WINDOW_WIDTH / 4, 13 * WINDOW_WIDTH / 16);const Point* ppt[1] = { rookPoints[0] };int npt[] = { 20 };fillPoly(img,ppt,npt,1,Scalar(255, 255, 255),lineType);
}//-----------------------------------【DrawLine( )函数】--------------------------
//      描述:自定义的绘制函数,实现了线的绘制
//---------------------------------------------------------------------------------
void DrawLine(Mat img, Point start, Point end)
{int thickness = 2;int lineType = 8;line(img,start,end,Scalar(0, 0, 0),thickness,lineType);
}

OpenCV3编程入门(毛星云)读书笔记(一)相关推荐

  1. 《OpenCV3编程入门-毛星云》第三部分 掌握imgproc组件

    平台:Win7 64bits + Visual Studio 2012 + OpenCV 2.4.10 截止今天我终于把<OpenCV3编程入门-毛星云>这本书看完了,看了将近两个月终于看 ...

  2. 图像识别(5)——《OpenCV3编程入门-毛星云》第三部分 掌握imgproc组件

    博主QQ:1356438802 QQ群:473383394--UVC&OPENCV473383394 平台:Win7 64bits + Visual Studio 2012 + OpenCV  ...

  3. 《OpenCV3编程入门-毛星云》第一部分 快速上手OpenCV

    平台:Win7 64bits + Visual Studio 2012 + OpenCV 2.4.10 接下来的很长一段时间我将沿着学习,记录整个学习过程和心得,既是自己的学习笔记,也为后来人提供一份 ...

  4. opencv3编程入门毛星云(第三章3.18-3.21)

    3.1.8输出图像: #include<opencv2/opencv.hpp> #include <stdio.h> #include<vector> using ...

  5. 图像识别(2)——《OpenCV3编程入门》毛星云编著

    博主QQ:1356438802 QQ群:473383394--UVC&OpenCV47 最近几天开始收集图像识别的相关资料: 程序开源库主要就是OpenCV,这是主流工具,所以我暂时也没去了解 ...

  6. OpenCV3编程入门(毛星云)之视频读取与播放

    OpenCV3编程入门(毛星云)之视频读取与播放,看看运行截图: 看来OpenCV API封装得太好了..努力,努力,再努力~~~

  7. 【OpenCV学习】 《OpenCV3编程入门》--毛星云 01 邂逅OpenCV(OpenCV基本概念与基本架构) ROS系统上的运用(python实现)

    对 <OpenCV3编程入门>第一章的学习笔记:理解什么是计算机视觉,什么是OpenCV,以及其中的联系等等. PS:此书为2014年出版,opencv的版本和接口也与现在有些不一致了,作 ...

  8. OpenCV3编程入门(毛星云)之用滚动条控制两图片的混合

    OpenCV3编程入门书本上的示例,copy下来学习学习~~代码如下: 美女1 美女2 #include <opencv2/opencv.hpp> using namespace cv; ...

  9. 【资源】OpenCV3编程入门_毛星云

    OpenCV在计算机视觉领域扮演着重要的角色.作为一个基于开源发行的跨平台计算机视觉库,OpenCV实现了图像处理和计算机视觉方面的很多通用算法.<OpenCV3编程入门>以当前最新版本的 ...

  10. 原创 OpenCV3编程入门 学习笔记(总)

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/qq_36163358/article/ ...

最新文章

  1. 免费报名 | 机器学习的第二次入门(入群有福利)
  2. 反模式:神仙大类和黄金大锤
  3. 数据库:Redis数据库优点介绍
  4. LeetCode 659. 分割数组为连续子序列(哈希)
  5. C++学习之路 | PTA乙级—— 1082 射击比赛 (20 分)(精简)
  6. 【HTML5】HTML5基础语法汇总
  7. myeclipse上SVN代码合并详细步骤图解
  8. 万兆安全网关选型指南
  9. PHP后端美化,基于Thinkphp5.5表白墙源码,已美化后端
  10. zktime 协议_ZKTiMe5.0考勤管理系统使用介绍(1.3版).pdf
  11. Ubuntu 安装显卡驱动 CUDA10 cuDNN详细教程
  12. 【虚拟机数据恢复】误删除VMware虚拟机vmdk文件的数据恢复案例
  13. 关于类的符号输入过程第二篇
  14. 服务器装win7无限重启吗,win7下机子无限重启解决办法
  15. 全基因组选择中准确性的影响因素
  16. “另类”计算IP地址和子网划分
  17. matlab怎么读txt文件字符串,Matlab中读取txt文件的几种方法
  18. 二本院校学弟大二开始实习,大三收割阿里、腾讯实习offer
  19. ElasticSearch——手写一个ElasticSearch分词器(附源码)
  20. html浅色背景,深色背景和浅色标签

热门文章

  1. html超链接图标图片,HTML-标签:图片 超链接
  2. setup factory的命令行打包
  3. 学以致用深入浅出数字信号处理 pdf_数字阵列雷达--相控阵专题讲座之三
  4. Mac_苹果电脑设置眼睛保护色
  5. 微信小游戏源码(从入门到入坑-火柴人勇闯地下城))
  6. 百度云盘群组中资源文件实时同步更新保存到自己群组的方法
  7. 怎么更改AD域用户账号和密码_AD域管理中那些实用的软件
  8. 高频实验设备,高频电子线路信号发生器实验箱
  9. 解决安卓4.4webview的兼容性问题
  10. vue滑块滑动校验,兼容移动端/pc端