OpenCV的简单使用教程(C++)
-
- OpenCV简介
- OpenCV的使用基础
- 打开、显示和保存图像
- 图像存储变量 Mat类
- 图像元素的存储
- 读入图像文件
- 创建Mat类
- 复制Mat类
- 图像元素的访问
- OpenCV画图
- 命令行交互界面
OpenCV简介 OpenCV是一个开源发行的跨平台计算机视觉和机器学习软件库,提供了图像处理和计算机视觉方面的很多通用算法。接下来,我简要介绍一些OpenCV基础且常用的函数(C++接口)。
OpenCV的使用基础 在这里,我就不讲OpenCV的配置问题了。在使用OpenCV接口时,记得在代码开头的预处理命令部分加入:
#include
#include
为了方便和编写可读性高、简洁的代码,墙裂建议使用命名空间:
using namespace cv;
// 省去函数前面加cv::的必要性
当然,也可以不使用命名空间,但是在使用时需自己写cv,例如:
cv::Mat img;
打开、显示和保存图像 打开图像:imread 显示图像:imshow 保存图像:imwrite 例如:
int main()
{
char imageName[] = "/test.jpg";
Mat M=imread(imageName,IMREAD_COLOR);
// 读入图片 if(M.empty())// 判断文件是否正常打开
{
fprintf(stderr, "Can not load image %s\n", imageName);
waitKey(6000);
// 等待6000 ms后窗口自动关闭
return -1;
}imshow("image",M);
// 显示图片
waitKey();
imwrite("pic.bmp",M);
// 存为bmp格式图片
return 0;
}
图像存储变量 Mat类 Mat类包括两个数据部分
(1)矩阵头:大小恒定,包括矩阵大小、存储方法、矩阵存储地址等信息
(2)矩阵指针:大小不定,指向包含了像素值的矩阵(可选择存储方法,采用任何维度存储数据)
// Mat的相关参数
int depth = M.depth();
//元素数据类型,枚举型,0为CV_8U
int channels = M.channels();
//色彩通道数
int nRows = M.rows;
//行数,图像高度
int nCols = M.cols;
//列数,图像宽度
uchar * data = https://www.it610.com/article/M.data;
//图像地址
int nDims = M.dims;
//图像维度
int nElSize = M.elemSize();
//每个像素字节数
int nElSize1 = M.elemSize1();
//每个像素单通道的字节数
int type = M.type();
//元素数据类型及通道数,枚举型
Type:枚举类型
type的命名格式为CV_(位数)+(数据类型)+(通道数)
文章图片
图像元素的存储 1.单通道灰度图 (像素通常为< uchar >类型)
文章图片
2.RGB三通道彩色图 (像素通常为< Vec3b >类型)
文章图片
注意事项:①行号、列号都是从“0”开始;②RGB彩色图的通道顺序反过来了,在OpenCV中是BGR。
读入图像文件 读入图像使用imread函数,用法如下:
Mat M;
//只构造了Mat类的矩阵头
M = imread(argv[1], IMREAD_COLOR);
// 开辟矩阵空间,并将文件的内容写入函数声明
Mat imread(const string& filename, int flags);
IMREAD_UNCHANGED-1
IMREAD_GRAYSCALE0
IMREAD_COLOR1
IMREAD_ANYDEPTH2
IMREAD_ANYCOLOR4
缺省值为1
创建Mat类 使用Mat函数可以创建图像矩阵,并可指定大小
// 利用成员函数 create() 创建图像矩阵
Mat M;
M.create(4,4, CV_8UC2);
//4行4列,开辟内存,但未指定像素的初值
cout << "M = "<< endl << " " << M << endl << endl;
// 利用Mat类的构造函数创建图像矩阵
//2维图像—平面图像
Mat M(2,2, CV_8UC3, Scalar(0,0,255));
//2行2列,给定全体像素的值
cout << "M = " << endl << " " << M << endl << endl;
system(" pause ");
//用于让命令窗口停留显示//3维图像—立体图像
(构造3维矩阵时,需先定义一个数组,说明每个维度的尺寸)
int sz[3] = {3,3,3};
//每个维度尺寸都是3
Mat M(3,sz,CV_8UC1,Scalar::all(0));
//高维无法直接用cout显示//构造指定内容的特殊矩阵
// 用类似matlab的特殊矩阵为第一个通道的矩阵赋值
Mat M = Mat::eye(4, 4, CV_64FC2);
Mat M = Mat::ones(2, 2, CV_32F);
Mat M = Mat::zeros(3,3, CV_8UC1);
//指定每个像素
Mat M = (Mat_(3,3) << 0, -1, 0, -1, 5, -1, 0, -1, 0);
Mat_ 对应 CV_64F;Mat_对应 CV_8U
Mat_对应 CV_8S; Mat_对应 CV_32S
Mat_对应 CV_32F; Mat_对应 CV_64F//利用随机数给矩阵赋值
Mat M = Mat(3, 2, CV_8UC3);
cv::randu(M, Scalar::all(0), Scalar::all(255));
//上限255,下限0//利用外部数据指针赋值
Mat M(height, width, CV_8UC3, pixels);
//char * pixels;
复制Mat类 当需要复制图像或者生成一张与原图像大小的新图像时,要用使用复制Mat类
//仅复制矩阵头,不复制数据
//共享全部数据
Mat B(A);
// 将A的矩阵头复制给B
C = A;
// 将A的矩阵头复制给C//共享部分数据
Mat D(A, Rect(0,0,2,3));
//D为A中以(0,0)为左上角,宽2,高3的矩形部分
Mat E = A(Range::all(),Range(0,3));
//E包括A的每一行,及其中第0列到第2列//复制矩阵头,且复制一份新数据
//克隆
Mat F = A.clone();
//拷贝
Mat G;
A.copyTo(G);
图像元素的访问 因为灰度图像和彩色图像的通道数不一样,所以两者的访问方式不一样,需要注意。下面我主要讲解彩色图像元素的访问(三通道),灰度图像的访问只需要单通道和用Mat.at< uchar >(j,i)访问即可。
访问方式有三种:data指针、Mat.at(i,j)和Mat.ptr行指针
//1. 使用data指针(unsigned char*)加偏移量
unsigned char* ptr=M.data;
int height = M.rows;
//number of rows
int width = M.cols;
//number of colums
for (int i=0;
i
//2. 用Mat.at(i,j)访问坐标(i,j)的图像元素
int height = M.rows;
//number of rows
int width = M.cols;
//number of colums
for (int i=0;
i(i,j)[0] = M.at(i,j)[0]*i*j/(height*width);
M.at(i,j)[1] = M.at(i,j)[1]*i*j/(height*width);
M.at(i,j)[2] = M.at(i,j)[2]*i*j/(height*width);
}
}
int height = M.rows;
//number of rows
int width = M.cols;
//number of colums
Vec3b* p;
for (int i=0;
i(i);
for (int j=0;
j
OpenCV画图 使用OpenCV可以画出很多的基本图形,应有尽有,比如:画圆、画直线、插入文字等。
// 画圆
Point center = Point(255,255);
//圆心
int r = 100;
//半径
circle(M,center,r,Scalar(0,0,0));
//参数为:原图像、圆心、半径、颜色、粗细、线型
imshow(“显示图像",img);
// 画椭圆弧线
Size s=Size(20,10);
ellipse(M,center,s,0,10,180,Scalar(0,0,0));
//参数为:原图像、圆心、长短轴、径向夹角(水平面到长轴的夹角)、起始角度(长轴到起始边沿的夹角)、结束角度(长轴到结束点的夹角)、倾斜的矩形(可选项)、颜色、粗细、线性、偏移// 画线
Point a = Point(600,600);
line(M,a,center,Scalar(255,0,0));
//参数为:原图像、起始点、结束点、颜色、粗细、线型// 画矩形
rectangle(M,a,center,Scalar(255,0,0));
//参数为:原图像、顶点、对角点、颜色、粗细、大小// 插入文字
string words= "good luck";
putText(M,words,Point(M.rows/2,M.cols/4),CV_FONT_HERSHEY_COMPLEX,1,Scalar(255,0,0));
//参数为:承载的图片,插入的文字,文字的位置(文本框左下角),字体,大小,颜色
命令行交互界面 OpenCV提供了相关函数,有助于编写交互程序
带输入参数的可执行命令
//主函数增加两个入口参数
int main(int argc,char*argv[])
// argv[0]为命令自身,其后依次为输入参数;argc为数组的元素个数
对于int _tmain(int argc, _TCHAR* argv[])//C++版本的main函数
需设值项目属性的general?character set?Use Multi-Byte Character Set//调用变量时
Mat img=imread(argv[1],IMREAD_ANYCOLOR);
// 调试时
在Solution Explore中右键项目(或选中项目后点project菜单)
Properties --> Configuration Properties -> Debugging
在Command Arguments中,输入入口参数(如C:\news.jpg),参数间以空格隔开//命令窗口调用时
打开项目的debug目录(xxx.exe所在目录)
空白处“Shift+右键”,选择“在此处打开命令窗口”
输入项目名及参数,如 xxx C:\news.jpg执行过程中的命令行输入交互
// 加入命令行输入提示
cout<<"请选择操作 [1] xx;
[2] xx\n";
// 读入选择
int x;
cin>>x;
// 根据选择,提示下一步输入,及执行相应操作
string imagename;
if (x == 1)
{cout<<"请输入图片文件名“<>imagename;
Mat img=imread(imagename,IMREAD_COLOR);
….
}
更多详细的OpenCV函数说明,请移步官方文档:
https://docs.opencv.org/3.4/d6/d00/tutorial_py_root.html
推荐阅读
- Python从零到壹|[Python从零到壹] 四十三.图像增强及运算篇之图像点运算和图像灰度化处理
- 《C++要笑着学》|【C++要笑着学】认识C++ | C++的发展史 | 学习方法建议
- 《C++要笑着学》|【C++要笑着学】缺省参数 | 函数重载
- 《C++要笑着学》|【C++要笑着学】内联函数 | 关键字auto | 范围for | 关键字 nullptr
- C/C++编程(libcurl学习(linux + cmake))
- C++一笑而过|C++学习之第十三天-移动语义与完成COW String类
- 指针|【发际线大作战】C++学习记录之用户自定义数据类型
- 小问C++笔记|C++之Big Three(拷贝构造、拷贝赋值、析构函数探究)
- 【发际线大作战】C++学习记录之数组(发际线-1cm)