OpenCV 入门级一
首先声明,这一系列博文是本人在学习 OpenCV 的过程中对自身学习的一个总结而已,
仅作为 OpenCV 入门级参考,并没有什么内容值得高手参考,
所以,只适合和我一样正值学习状态者阅读,高手无意者请飘过,但欢迎留言指教。
OpenCV 是一个开源的计算机视觉库,其采用 C/C++ 编写,被设计为可移植的库,
OpenCV 的设计目标是执行速度尽可能的快,
其主要关注的是实时应用,同时,OpenCV 的另一个目标是构建一个简单易用的计算机视觉框架,
以帮助开发人员更便捷地设计更复杂的计算机视觉相关的应用程序。
OpenCV 的结构和内容
上面这幅截图是 OpenCV 源码的文件组成结构,可以看出其中包括 cv ,cvaux , cxcore , highgui , ml 这 5 个模块。
如果以库来体现这几个模块之间的关系的话,可以采用下面的结构图来展现:
CV: 包含了基本的图像处理函数和高级的计算机视觉算法,包括图像处理,
图像结构分析,运动描述和跟踪,模式识别和摄像机标定。
ML: 是机器学习库,包含一些基于统计的分类和聚类工具。
HighGUI: 包含图像和视频的输入/输出函数。
CXCORE: 包含了 OpenCV 的一些基本的数据结构和相关函数。
CVAUX: 该模块则是一般用存放即将被淘汰的算法和函数,同时也包含一些新出现的实验性的函数和算法。
关于 OpenCV 的介绍到此介绍,下面呢,就是通过一个 Demo 来学习一些常用的 API 的使用。
Demo 学习
Demo01
下面的 Demo 将介绍如何显示一张图片,具体 API 的使用请注意注释。
//在HighGUI 模块中包含了图像和视频的输入/输出的基本函数 #include "highgui.h" //使用了命令行下运行的形式,其中要在命令行下传递参数 int main(int argc,char ** argv) { //通过cvLoadImage 将一幅指定路径的图片加载到内存中 //同时会生成一个IplImage 类型的结构 //这个结构会指向图片所在的内存区域 IplImage * image=cvLoadImage(argv[1]); //通过cvNamedWindow 来建立一个窗体 //因为图片必须在指定的窗体中才能显示 cvNamedWindow("Demo01",0); //通过cvShowImage 函数来指定在指定的窗口中显示指定的图片 cvShowImage("Demo01",image); //暂停程序的执行 //只有当用户按下任意键后才执行后面的代码 cvWaitKey(0); //释放掉加载到内存中的图片所占的内存资源 cvReleaseImage(&image); //销毁窗口 cvDestroyWindow("Demo01"); }
至于程序的执行的话,必须到命令行下执行,同时还需要将一张图片拷贝到 exe 文件所在的目录,
然后再在命令行中指定该图片作为参数传递到 Main 函数中。
Demo02
下面继续看下一个 Demo,这个 Demo 将展示播放视频文件:
//图像和视频的输入/输出均在HighGUI 模块中 #include "highgui.h" //播放视频文件只需要循环的顺序的读取视频中的每一帧 //读到帧后,便可以将这个帧作为普通的图像一样显示即可 int main(int argc,char **argv) { //首先需要建立一个窗口来容纳视频的播放 cvNamedWindow("Demo02",CV_WINDOW_AUTOSIZE); //打开一个视频文件,返回的 CvCapture 结构中包含了视频文件的信息 CvCapture *capture=cvCreateFileCapture(argv[1]); IplImage *frame; //循环顺序的读取视频中的帧 while(1) { //获取当前播放帧的下一个帧,并且将获取到的帧加载到内存中,覆盖掉前面帧所占的内存控件 frame=cvQueryFrame(capture); if(!frame) { //如果没有读取到帧的话则说明播放完毕了 //从而退出播放 break; } //将读取到的帧显示在窗口中 cvShowImage("Demo02",frame); //每播放一个帧就在此等待30 毫秒 char c=cvWaitKey(30); //如果在30 ms 中用户按下了ESC 键 //(ESC 的ASCII 为27)则退出播放 if(c==27) { break; } } //分配的内存需要手动释放 cvReleaseCapture(&capture); //销毁窗口 cvDestroyWindow("Demo02"); }
Demo03
上面呢,确实是可以成功的播放视频了,但是一般的视频播放器都是有个滚动条的,
允许用户手动拖动滚动条从而定位到视频指定的帧上,
下面我们就要实现这个功能了。
//CV 模块中包含了图像处理,图像结构分析, //运动描述和跟踪,模式识别和摄像机标定 #include "cv.h" //包含了图像和视频的输入/输出 #include "highgui.h" //用来标定滚动条当前的位置 int g_Pos=0; //将打开的视频文件作为一个全局变量使用 CvCapture * g_Capture=NULL; //当拖动滚动条后,会回调这个函数 //同时会将当前滚动条的位置以32 位形式传递过来 void CallBackTrackBarSlide(int pos) { //这里便是重新设置视频文件当前播放的帧 cvSetCaptureProperty(g_Capture,CV_CAP_PROP_POS_FRAMES,pos); } int main(int argc,char **argv) { cvNamedWindow("Demo03",CV_WINDOW_AUTOSIZE); //根据参数打开指定的视频文件 g_Capture=cvCreateFileCapture(argv[1]); //获得总的帧数 int totalFrames= (int)cvGetCaptureProperty(g_Capture,CV_CAP_PROP_FRAME_COUNT); if(totalFrames!=0) { //创建滚动条,在这里指定了滚动条拖动后的回调函数 cvCreateTrackbar("TrackBar","Demo03", &g_Pos,totalFrames,CallBackTrackBarSlide); } IplImage * frame; //循环的顺序的遍历所有的帧 while(1) { //获取当前帧的下一个帧,并将其加载到内存中 frame=cvQueryFrame(g_Capture); if(!frame) { break; } cvShowImage("Demo03",frame); char chKeyCode=cvWaitKey(30); if(chKeyCode==27) { break; } } cvReleaseCapture(&g_Capture); cvDestroyWindow("Demo03"); return 0; }
在测试的时候需要注意,有一些视频文件时不支持动态指定帧的操作,所以有可能拖动时会失败。
然后就可以拖动滚动条来定位帧了
Demo03
上面的 Demo 呢确实是实现了可以通过拖动滚动条来实现对视频帧的动态控制,
但是有一个问题就是滚动条并不会跟随视频的播放而自带增加,也就是,随时时间流逝,
视频会一直播放,但是滚动条如果不人为地拖动的话是不会发生改变的,
而我们要是实现的就是当视频播放到哪一个帧了,滚动条就应该位于相应的位置上,
所以下面的 Demo 就来实现这个功能。
//CV 模块中包含了图像处理,图像结构分析, //运动描述和跟踪,模式识别和摄像机标定 #include "cv.h" //包含了图像和视频的输入/输出 #include "highgui.h" //用来标定滚动条当前的位置 int g_Pos=0; //将打开的视频文件作为一个全局变量使用 CvCapture * g_Capture=NULL; //当拖动滚动条后,会回调这个函数 //同时会将当前滚动条的位置以32 位形式传递过来 void CallBackTrackBarSlide(int pos) { //这里便是重新设置视频文件当前播放的帧 cvSetCaptureProperty(g_Capture,CV_CAP_PROP_POS_FRAMES,pos); g_Pos=pos; } int main(int argc,char **argv) { //建立一个名字叫做Demo04 的窗体 cvNamedWindow("Demo04",CV_WINDOW_AUTOSIZE); //根据参数打开指定的视频文件 g_Capture=cvCreateFileCapture(argv[1]); //获得总的帧数 int totalFrames= (int)cvGetCaptureProperty(g_Capture,CV_CAP_PROP_FRAME_COUNT); if(totalFrames!=0) { //创建滚动条,在这里指定了滚动条拖动后的回调函数 cvCreateTrackbar("TrackBar","Demo04", &g_Pos,totalFrames,CallBackTrackBarSlide); } IplImage * frame; //循环的顺序的遍历所有的帧 while(1) { //获取当前帧的下一个帧,并将其加载到内存中 frame=cvQueryFrame(g_Capture); if(!frame) { break; } cvShowImage("Demo04",frame); char chKeyCode=cvWaitKey(30); if(chKeyCode==27) { break; } g_Pos++; //当播放完200 个帧的时候才触发滚动条滚动 if(g_Pos%200==0) { cvSetTrackbarPos("TrackBar","Demo04", g_Pos); } } cvReleaseCapture(&g_Capture); cvDestroyWindow("Demo03"); return 0; }
从而实现了滚动条随着视频的播放而自动滚动的效果。
这一篇博文就写到这里了,从上面可以看出,主要是随着逐步的深入来介绍一些 OpenCV 的常用的 API ,
其主线是首先是加载一张图片,而后再是加载视频,再在视频中加入滚动条,最后是滚动条和视频联动的效果,
随着一步一步的深入,可以更好的熟悉 OpenCV 的几个常用的 API 。
2010 年 10 月 23 日