SIFT算法实现C语言.pdf
《SIFT算法实现C语言.pdf》由会员分享,可在线阅读,更多相关《SIFT算法实现C语言.pdf(34页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、经典算法 SIFT 实现即代码解释:以下便是 sift 源码库编译后的效果图:为了给有兴趣实现sift 算法的朋友提供个参考,特整理此文如下。要了解什么是 sift 算法,请参考:九、图像特征提取与匹配之SIFT 算法。ok,咱们下面,就来利用 Rob Hess维护的 sift 库来实现 sift 算法:首先,请下载 Rob Hess维护的 sift 库:http:/blogs.oregonstate.edu/hess/code/sift/下载 Rob Hess的这个压缩包后,如果直接解压缩,直接编译,那么会出现下面的错误提示:编 译提 示:error C1083:Cannot open in
2、clude file:cxcore.h:No such file or directory,找不到这个头文件。这个错误,是因为你还没有安装opencv,因为:cxcore.h和 cv.h 是开源的 OPEN CV 头文件,不是 VC+的默认安装文件,所以你还得下载OpenCV 并进行安装。然后,可以在 OpenCV 文件夹下找到你所需要的头文件了。据网友称,截止 2010年 4 月 4 日,还没有在 VC6.0 下成功使用 opencv2.0的案例。所以,如果你是 VC6.0 的用户请下载 opencv1.0版本。vs 的话,opencv2.0,1.0任意下载。以下,咱们就以 vc6.0 为平
3、台举例,下载并安装 opencv1.0版本、gsl 等。当然,你也可以用 vs 编译,同样下载 opencv(具体版本不受限制)、gsl 等。请按以下步骤操作:一、下载 opencv1.0 http:/ opencv1.0,配置 Windows 环境变量 1、安装注意:假如你是将OpenCV 安装到 C:/Program Files/OpenCV(如果你安装的时候选择不是安装在C 盘,则下面所有对应的C 盘都改为你所安装在的那个“X盘”,即可),在安装时选择 将/OpenCV/bin 加入系统变量,打上“勾”。(Add/OpenCV/bin to the systerm PATH。这一步确认选
4、上了之后,下面的检查环境变量的步骤,便可免去)2、检查环境变量。为了确保上述步骤中,加入了系统变量,在安装opencv1.0成功后,还得检查C:/Program Files/OpenCV/bin 是否已经被加入到环境变量PATH,如果没有,请加入。3、最后是配置 Visual C+6.0。全局设置菜单 Tools-Options-Directories:先设置 lib 路径,选择 Library files,在下方填入路径:C:/Program Files/OpenCV/lib 然后选择 include files,在下方填入路径(参考下图):C:/Program Files/OpenCV/c
5、xcore/include C:/Program Files/OpenCV/cv/include C:/Program Files/OpenCV/cvaux/include C:/Program Files/OpenCV/ml/include C:/Program Files/OpenCV/otherlibs/highgui C:/Program Files/OpenCV/otherlibs/cvcam/include 最后选择 source files,在下方填入路径:C:/Program Files/OpenCV/cv/src C:/Program Files/OpenCV/cxcore/
6、src C:/Program Files/OpenCV/cvaux/src C:/Program Files/OpenCV/otherlibs/highgui C:/Program Files/OpenCV/otherlibs/cvcam/src/windows 项目设置每创建一个将要使用OpenCV 的 VC Project,都需要给它指定需要的lib。菜单:Project-Settings,然后将 Setting for 选为 All Configurations,然后选择右边的 link 标签,在 Object/library modules 附加上:cxcore.lib cv.lib
7、ml.lib cvaux.lib highgui.lib cvcam.lib 当然,你不需要这么多lib,你可以只添加你需要的lib(见下图)三、下载 gsl,gsl 也是一个库,也需要下载:http:/ GSL 也是和 OpenCV 一样要把头文件和lib 的路径指定好。四、配置 gsl 将 C:/WinGsl/bin 中的 WinGsl.dll 和 WinGslD.dll 复制到 C:/VC6.0/Bin;将整个Gsl 目录 复制 到 C:/VC6.0/Bin下;lib 目 录下的所有.lib文件 全部复制到C:/VC6.0/Lib 下。然后,在 toolsoptionsdirectori
8、es 中,将 C:/WinGsl 下的 lib,gsl 分别加入到库文件和头文件的搜索路径中。以下是可能会出现的错误情况处理:I、OpenCV 安装后“没有找到 cxcore100.dll”的错误处理在安装时选择“将/OpenCV/bin 加入系统变量”(Add/OpenCV/bin to the systerm PATH)。但该选项并不一定能成功添加到系统变量,如果编写的程序在运行时出现“没有找到cxcore100.dll,因为这个应用程序未能启动。重新安装应用程序可能会修复此问题。”的错误。手动在我的电脑-属性-高级-环境变量-系统变量-path 添加 c:/program files/o
9、pencv/bin;添加完成后需要重启计算机。II、vc6.0 下配置了一下,可是编译程序时遇到如下一个错误:Linking.LINK:fatal error LNK1104:cannot open fileodbccp32.libcxcore.lib 可能是:在工程设置的时候添加连接库时没加空格或.来把两个文件名(odbccp32.lib cxcore.lib)分开。注意每一次操作后,记得保存。若经过以上所有的步骤之后,如果还不能正常编译,那就是还要稍微修改下你下载的 Rob Hess代码。ok,日后,若有空,再好好详细剖析下此sift 的源码。最后,祝你编译顺利。完。SIFT 代码详解:这
10、是一个很强大的算法,主要用于图像配准和物体识别等领域,但是其计算量相比也比较大,性价比比较高的算法包括PCA-SIFT 和 SURF 其中 OpenCV 提供了 SURF 算法,但是为了方便理解。这里给出了Rob Hess 所实现的 SIFT 算法的实现以及注释,结合我自己的理解,如果,您有关于SIFT 算法不理解的地方咱们可以一起交流一下。或者您认为不详细的地方提出来。SIFT 算法的主要实现在sift.c 这个文件,其主要流程为:(1)首先创建初始图像,即通过将图像转换为32 位的灰度图,然后将图像使用三次插值来方大,之后通过高斯模糊处理(2)在此基础上进行高斯金字塔的构建以及高斯差分金字
11、塔的构建(3)对图像进行极值点检测(4)计算特征向量的尺度(5)调整图像大小(6)计算特征的方向(7)计算描述子,其中包括计算二维方向直方图并转换直方图为特征描述子首先给出 sift 算法的整体框架代码:输入参数:img 为输入图像;feat 为所要提取的特征指针;intvl 指的是高斯金字塔和差分金字塔的层数;sigma 指的是图像初始化过程中高斯模糊所使用的参数;contr_thr是归一化之后的去除不稳定特征的阈值;curv_thr指的是去除边缘的特征的主曲率阈值;img_dbl是是否将图像放大为之前的两倍;descr_with用来计算特征描述子的方向直方图的宽度;descr_hist_b
12、ins是直方图中的条数cppview plaincopy1.int _sift_features(IplImage*img,struct feature*feat,int intvls,2.double sigma,double contr_thr,int curv_thr,3.int img_dbl,int descr_width,int descr_hist_bins)4.5.IplImage*init_img;6.IplImage*gauss_pyr,*dog_pyr;7.CvMemStorage*storage;8.CvSeq*features;9.int octvs,i,n=0;10
13、.11./*check arguments*/12.if(!img)13.fatal_error(NULL pointer error,%s,line%d,_FILE_,_LINE_);14.15.if(!feat)16.fatal_error(NULL pointer error,%s,line%d,_FILE_,_LINE_);17.18./*build scale space pyramid;smallest dimension of top level is 4 pixels*/19./*构建高斯尺度空间金字塔,顶层最小的为4像素 */20.init_img=create_init_i
14、mg(img,img_dbl,sigma);21.octvs=log(double MIN(init_img-width,init_img-height)/log(2.0)-2;22./构建高斯金字塔和高斯差分金字塔23.gauss_pyr=build_gauss_pyr(init_img,octvs,intvls,sigma);24.dog_pyr=build_dog_pyr(gauss_pyr,octvs,intvls);25.26.storage=cvCreateMemStorage(0);27.28./尺度空间极值点检测29.features=scale_space_extrema(d
15、og_pyr,octvs,intvls,contr_thr,30.curv_thr,storage);31.32./画出去除低对比度的极值点33./draw_extrempoint(img,features);34.35.36.37.38./计算特征向量的尺度39.calc_feature_scales(features,sigma,intvls);40.if(img_dbl)41.adjust_for_img_dbl(features);42./计算特征的方向43.calc_feature_oris(features,gauss_pyr);44./计算描述子,包括计算二维方向直方图和转换其为
16、特征描述子pute_descriptors(features,gauss_pyr,descr_width,descr_hist_bins);46.47./*sort features by decreasing scale and move from CvSeq to array*/48.cvSeqSort(features,(CvCmpFunc)feature_cmp,NULL);49.n=features-total;50.*feat=static_cast(calloc(n,sizeof(struct feature);51.*feat=static_cast(cvCvtSeqToArr
17、ay(features,*feat,CV_WHOLE_SEQ);52.53.54.55.56.for(i=0;i width*2,img-height*2),11.IPL_DEPTH_32F,1);12.cvResize(gray,dbl,CV_INTER_CUBIC);13.cvSmooth(dbl,dbl,CV_GAUSSIAN,0,0,sig_diff,sig_diff);14.cvReleaseImage(&gray);15.return dbl;16.17.else18.19.sig_diff=sqrt(sigma*sigma-SIFT_INIT_SIGMA*SIFT_INIT_SI
18、GMA);20.cvSmooth(gray,gray,CV_GAUSSIAN,0,0,sig_diff,sig_diff);21.return gray;22.23.(2)构建高斯金字塔输入参数:octvs 是高斯金字塔的组invls 是高斯金字塔的层数sigma 是初始的高斯模糊参数,后续也通过它计算每一层所使用的sigmacppview plaincopy1.static IplImage*build_gauss_pyr(IplImage*base,intoctvs,int intvls,double sigma)2.3.IplImage*gauss_pyr;4.double*sig=st
19、atic_cast(calloc(intvls+3,sizeof(double);5.double sig_total,sig_prev,k;6.int i,o;7.8.gauss_pyr=static_cast(calloc(octvs,sizeof(IplImage*);9.for(i=0;i octvs;i+)10.gauss_pyri=static_cast(calloc(intvls+3,sizeof(IplImage*);11.12./*13.precompute Gaussian sigmas using the following formula:14.预计算每次高斯模糊的si
20、gma15.16.sigma_total2=sigma_i2+sigma_i-1217.*/18.sig0=sigma;19.k=pow(2.0,1.0/intvls);20.for(i=1;i intvls+3;i+)21.22.sig_prev=pow(k,i-1)*sigma;23.sig_total=sig_prev*k;24.sigi=sqrt(sig_total*sig_total-sig_prev*sig_prev);25.26.27.28.for(o=0;o octvs;o+)29.for(i=0;i intvls+3;i+)30.31./对每一层进行降采样,形成高斯金字塔的每
21、一层32.if(o=0&i=0)33.gauss_pyroi=cvCloneImage(base);34.35./*base of new octvave is halved image from end of previous octave*/36./每一组的第一层都是通过对前面一组的第一层降采样实现的37.elseif(i=0)38.gauss_pyroi=downsample(gauss_pyro-1intvls);39.40./*blur the current octaves last image to create the next one*/41./每一组的其他层则使通过使用不同
22、sigma 的高斯模糊来进行处理42.else43.44.gauss_pyroi=cvCreateImage(cvGetSize(gauss_pyroi-1),45.IPL_DEPTH_32F,1);46.cvSmooth(gauss_pyroi-1,gauss_pyroi,47.CV_GAUSSIAN,0,0,sigi,sigi);48.49.50.51.free(sig);52.return gauss_pyr;53.降采样处理输入参数:不解释这就是降采样,其实就是将图像通过最近邻算法缩小为原来的一半cppview plaincopy1.static IplImage*downsample
23、(IplImage*img)2.3.IplImage*smaller=cvCreateImage(cvSize(img-width/2,img-height/2),4.img-depth,img-nChannels);5.cvResize(img,smaller,CV_INTER_NN);6.7.return smaller;8.(3)构建高斯差分金字塔输入参数:不解释了参见上面的说明即可实际上差分金字塔的构成是通过对相邻层的图像进行相减获得的cppview plaincopy1.static IplImage*build_dog_pyr(IplImage*gauss_pyr,int octv
24、s,int intvls)2.3.IplImage*dog_pyr;4.int i,o;5.6.dog_pyr=static_cast(calloc(octvs,sizeof(IplImage*);7.for(i=0;i octvs;i+)8.dog_pyri=static_cast(calloc(intvls+2,sizeof(IplImage*);9.10.for(o=0;o octvs;o+)11.for(i=0;i intvls+2;i+)12.13.dog_pyroi=cvCreateImage(cvGetSize(gauss_pyroi),14.IPL_DEPTH_32F,1);1
25、5.cvSub(gauss_pyroi+1,gauss_pyroi,dog_pyroi,NULL);16.17.18.return dog_pyr;19.(4)极值点检测输入参数:contr_thr 是去除对比度低的点所采用的阈值curv_thr 是去除边缘特征的阈值cppview plaincopy1.static CvSeq*scale_space_extrema(IplImage*dog_pyr,int octvs,int intvls,2.double contr_thr,int curv_thr,3.CvMemStorage*storage)4.5.CvSeq*features;6.
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- SIFT 算法 实现 语言
限制150内