Harris角点学习[通俗易懂]

Harris角点学习[通俗易懂]人们普遍认为角点是二维图像亮度变化剧烈的点或图像边缘曲线上曲率极大值的点。这些点在保留图像图形重要特征的同时,可以有效地减少信息的数据量,使其信息的含量很高,有效地提高了计算的速度,有利于图像的可靠匹配,使得实时处理成为可能。其在三维场景重建、运动估计、目标跟踪、目标识别、图像配准与匹配等计算机视觉领域起着非常重要的作用。  角点的检测主要有两类基于图像边缘的方法和基于图像灰度的方法。前者很

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

    人们普遍认为角点是二维图像亮度变化剧烈的点或图像边缘曲线上曲率极大值的点。这些点在保留图像图形重要特征的同时,可以有效地减少信息的数据量,使其信息的含量很高,有效地提高了计算的速度,有利于图像的可靠匹配,使得实时处理成为可能。其在三维场景重建、运动估计、目标跟踪、目标识别、图像配准与匹配等计算机视觉领域起着非常重要的作用。

   角点的检测主要有两类基于图像边缘的方法和基于图像灰度的方法。前者很大程度上依赖于图像的分割和边缘提取,一旦待检测目标发生局部变化,很可能导致操作失败,因此该类方法使用范围较小;后者有很多方法,包括Harris算子,Moravec算子,Susan算子等等。

  本文的目的是在学习了Harris算子后,进行一个总结。文献:《A COMBINED CORNER AND EDGE DETECTOR》,1988,Chris Harris & Mike Stephens

一、角点边缘的直观概念:

角点:最直观的印象就是在水平和竖直两个方向变化均较大的两个点,即X,Y方向梯度都较大

边缘:仅在水平或竖直方向有较大的变化量,即X、Y方向梯度只有一个较大

平坦区域:在水平和竖直方向的变化量均较小,即X、Y方向梯度都较小

下图是理想情况的角点、边缘和平坦区域的示意图:

Harris角点学习[通俗易懂]

二、原理:

  由于CSDN不知道怎么把编辑的公式弄进来,所以整段粘帖论文了,英文不难懂:

Harris角点学习[通俗易懂]

Harris角点学习[通俗易懂]

Harris角点学习[通俗易懂]

论文后面那三个注意点讲得不够详细而且要求特征值。1988年的论文中提出,可以用以下方法代替求特征值,并给出了角点相应函数R:

Harris角点学习[通俗易懂]

其中,A和B分别表示C(x,y)矩阵主对角线两个元素,C代表其副对角线元素。k和较大值判断的阈值T,在下面代码中给出,为经验数值。

如下图所示,当R为较大正数时,该区域为角点区域,当R为较大负数时,该区域为边缘区域,当R的绝对值较小时,该区域为平坦区域。

Harris角点学习[通俗易懂]

三、具体实现

我将分别给出matlab和 c 两个版本。其中,c 版本不是使用什么goodfeaturetotrack来实现的

matlab版本:

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%   Harris角点提取算法                                          %
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
clear;
%filename = 'Lena.jpg';
filename='xx.png';
X = imread(filename);     % 读取图像
% imshow(X);
Info = imfinfo(filename); %获取图像相关信息
if (Info.BitDepth > 8)
    f = rgb2gray(X);
end
%《基于特征点的图像配准与拼接技术研究》
%计算图像亮度f(x,y)在点(x,y)处的梯度-----------------------------------------------
% fx = [5 0 -5;8 0 -8;5 0 -5];          % 高斯函数一阶微分,x方向(用于改进的Harris角点提取算法)
ori_im = double(f) / 255;                   %unit8转化为64为双精度double64
fx = [-2 -1 0 1 2];                     % x方向梯度算子(用于Harris角点提取算法)
Ix = filter2(fx, ori_im);                % x方向滤波
% fy = [5 8 5;0 0 0;-5 -8 -5];          % 高斯函数一阶微分,y方向(用于改进的Harris角点提取算法)
fy = [-2; -1; 0; 1; 2];                     % y方向梯度算子(用于Harris角点提取算法)
Iy = filter2(fy, ori_im);                % y方向滤波
%构造自相关矩阵---------------------------------------------------------------
Ix2 = Ix .^ 2;
Iy2 = Iy .^ 2;
Ixy = Ix .* Iy;
clear Ix;
clear Iy;
h= fspecial('gaussian', [7 7], 2);        % 产生7*7的高斯窗函数,sigma=2
Ix2 = filter2(h,Ix2);
Iy2 = filter2(h,Iy2);
Ixy = filter2(h,Ixy);
%提取特征点---------------------------------------------------------------
height = size(ori_im, 1);
width = size(ori_im, 2);
result = zeros(height, width);           % 纪录角点位置,角点处值为1
R = zeros(height, width);
Rmax = 0;                              % 图像中最大的R值
k = 0.06; %k为常系数,经验取值范围为0.04~0.06
for i = 1 : height
    for j = 1 : width
        M = [Ix2(i, j) Ixy(i, j); Ixy(i, j) Iy2(i, j)];             % auto correlation matrix
        R(i,j) = det(M) - k * (trace(M)) ^ 2;                     % 计算R
        if R(i,j) > Rmax
            Rmax = R(i, j);
        end;
    end;
end;
%T = 0.01 * Rmax;%固定阈值,当R(i, j) > T时,则被判定为候选角点
T = 0.1 * Rmax;%固定阈值,当R(i, j) > T时,则被判定为候选角点

%在计算完各点的值后,进行局部非极大值抑制-------------------------------------
cnt = 0;
for i = 2 : height-1
    for j = 2 : width-1
        % 进行非极大抑制,窗口大小3*3
        if (R(i, j) > T && R(i, j) > R(i-1, j-1) && R(i, j) > R(i-1, j) && R(i, j) > R(i-1, j+1) && R(i, j) > R(i, j-1) && ...
                R(i, j) > R(i, j+1) && R(i, j) > R(i+1, j-1) && R(i, j) > R(i+1, j) && R(i, j) > R(i+1, j+1))
            result(i, j) = 1;
            cnt = cnt+1;
        end;
    end;
end;
i = 1;
    for j = 1 : height
        for k = 1 : width
            if result(j, k) == 1;
                corners1(i, 1) = j;
                corners1(i, 2) = k;
                i = i + 1;
            end;
        end;
    end;
[posc, posr] = find(result == 1);
figure,imshow(ori_im);
hold on;
plot(posr, posc, 'r+');

运行效果图:

Harris角点学习[通俗易懂]Harris角点学习[通俗易懂]

下面给出 c 语言版本,读图采用Opencv函数:

// My_corner.cpp : 定义控制台应用程序的入口点。//#include "stdafx.h"#include<iostream>#include <opencv2/core/core.hpp>  #include <opencv2/highgui/highgui.hpp>  #include <opencv2/imgproc/imgproc.hpp>#include <GL/glut.h>using namespace std;using namespace cv;void m_filter(double *src,double *&dst,int height,int width,double *mask,int mW,int mH){	double a;	for (int i = 0;i < height;i++)	{		for (int j = 0;j < width;j++)		{			a = 0.0;			//去掉靠近边界的行列,无法滤波,超出范围			if (i > int(mH/2) && i < height - int(mH/2) && j > int(mW) && j < width - int(mW/2))			{				for (int m = 0;m < mH;m++)				{					for(int n = 0;n < mW;n++)					{						a += src[(i+m-int(mH/2))*width+(j+n-int(mW))]*mask[m*mW+n];					}				}			}			dst[i*width+j] = a;		}	}}int _tmain(int argc, _TCHAR* argv[]){	Mat img_src = imread("test1.jpg");	Mat img_g,corner;	double *src,*img_x,*img_y,*img_x2,*img_y2,*img_xy;	Size size = img_src.size();	img_g = Mat(size,CV_8UC1);	corner = Mat(size,CV_8UC1);	src = new double[size.width*size.height];	img_x = new double[size.width*size.height];	img_y = new double[size.width*size.height];	img_x2 = new double[size.width*size.height];	img_y2 = new double[size.width*size.height];	img_xy = new double[size.width*size.height];	cvtColor(img_src,img_g,CV_BGR2GRAY);	for (int i=0;i<size.height;i++)	{		for (int j=0;j<size.width;j++)		{			src[i*size.width+j]=(double)img_g.data[i*size.width+j]/255;		}	}		//定义水平方向差分算子并求img_x	double dx[25] = {-2,-1,0,1,2,-2,-1,0,1,2,-2,-1,0,1,2,-2,-1,0,1,2,-2,-1,0,1,2};	m_filter(src,img_x,size.height,size.width,dx,5,5);	//定义垂直方向差分算子并求img_y	double dy[25] = {-2,-2,-2,-2,-2,-1,-1,-1,-1,-1,0,0,0,0,0,1,1,1,1,1,2,2,2,2,2};	m_filter(src,img_y,size.height,size.width,dy,5,5);	FILE *fp;	//fp=fopen("test.txt","w+");	//计算img_x2、img_y2、img_xy	for (int i = 0;i < size.height;i++)	{		for (int j = 0;j < size.width;j++)		{			img_x2[i*size.width+j] = img_x[i*size.width+j] * img_x[i*size.width+j];			img_y2[i*size.width+j] = img_y[i*size.width+j] * img_y[i*size.width+j];			img_xy[i*size.width+j] = img_x[i*size.width+j] * img_y[i*size.width+j];			//fprintf(fp,"%f,%f,%f,%f,%f",img_x[i*size.width+j],img_y[i*size.width+j],img_x2[i*size.width+j],img_y2[i*size.width+j],img_xy[i*size.width+j]);			//fprintf(fp,"\n");		}	}	//fclose(fp);	//对img_x2、img_y2、img_xy进行高斯平滑,本例中使用5×5的高斯模板	// 计算模版参数	int gausswidth = 7;	double sigma = 2;	double *g = new double[gausswidth*gausswidth];	for(int i = 0;i < gausswidth;i++)		for(int j = 0;j < gausswidth;j++)		{			g[i*gausswidth+j] = exp(-((i-int(gausswidth/2))*(i-int(gausswidth/2))+(j-int(gausswidth/2))*(j-int(gausswidth/2)))/(2*sigma));		}		//归一化:使模板参数之和为1	double total=0;	for(int i = 0;i < gausswidth*gausswidth;i++)		total += g[i];	for(int i =  0;i < gausswidth;i++)		for(int j = 0;j < gausswidth;j++){			g[i*gausswidth+j] /= total;		}	//进行高斯平滑	m_filter(img_x2,img_x2,size.height,size.width,g,gausswidth,gausswidth);	m_filter(img_y2,img_y2,size.height,size.width,g,gausswidth,gausswidth);	m_filter(img_xy,img_xy,size.height,size.width,g,gausswidth,gausswidth);	//计算角点量R,R(i,j) = det(M) - k * (trace(M)) ^ 2	//Tr(M)=img_x2+img_y2 ,det(M)=img_x2*img_y2-img_xy^2	double *R = new double[size.width*size.height];	double k = 0.06; //k为常系数,经验取值范围为0.04~0.06	double Rmax=0;	for(int i = 0; i < size.height; i++)	{		for(int j = 0; j < size.width; j++)		{			R[i*size.width+j] = img_x2[i*size.width+j] * img_y2[i*size.width+j] - img_xy[i*size.width+j] * img_xy[i*size.width+j] - k * (img_x2[i*size.width+j] + img_y2[i*size.width+j]) * (img_x2[i*size.width+j] + img_y2[i*size.width+j]);			if (R[i*size.width+j] > Rmax)			{				Rmax = R[i*size.width+j];			}			//printf("%f\n",R[i*size.width+j]);		}	}	double max;	double T = 0.1 * Rmax;//固定阈值,当R(i, j) > T时,则被判定为候选角点	double *mx = new double[size.width*size.height];	//在计算完各点的值后,进行局部非极大值抑制	for (int i=1;i<size.height-1;i++)	{		for (int j=1;j<size.width-1;j++)		{			//写不下了,分两行。。。进行非极大抑制,窗口大小3*3			if (R[i*size.width+j]> T && R[i*size.width+j] > R[(i-1)*size.width+j-1] && R[i*size.width+j] > R[(i-1)*size.width+j] && R[i*size.width+j] >R[(i-1)*size.width+j+1] && R[i*size.width+j] >R[i*size.width+j-1])			{				if (R[i*size.width+j]>R[i*size.width+j+1]&&R[i*size.width+j]>R[(i+1)*size.width+j-1]&&R[i*size.width+j]>R[(i+1)*size.width+j]&&R[i*size.width+j]>R[(i+1)*size.width+j+1])				{					corner.data[i*size.width+j]=1; 				}			}		}	}	//作图	Scalar sca(0,255,0);	for(int i = 0; i < size.height; i++)	{		for(int j = 0; j < size.width; j++)		{			if (corner.data[i*size.width+j] == 1)			{				Point p(j,i);				circle(img_src,p,20,sca);			}		}	}	imshow("find corner :)",img_src);	waitKey(0);	free(g);free(R);free(mx);	return 0;}

运行效果图:

Harris角点学习[通俗易懂]

Harris角点学习[通俗易懂]

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/189590.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 开源协议解读

    开源在今天的软件业已经很普遍,但开源是否意味着使用者可以对开源后的代码为所欲为呢?答案是否定的。开源运动同样有自己的游戏规则和道德准则。不遵行这些规则不但损害开源运动的健康发展,也会对违规者造成名誉和

    2021年12月22日
    58
  • 京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节「建议收藏」

    京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节「建议收藏」的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里,第京的季节里

    2022年10月3日
    0
  • SIGABRT错误的调试办法[通俗易懂]

    iOS经常会遇到一个头疼的error就是在main函数上显示“Thread1:signalSIGABRT”这个错误,终于在stackoverflow上找到了调试的办法: 原文链接:http://stackoverflow.com/questions/9782621/i-have-an-error-in-main-m-thread-1-signal-sigabrt-how

    2022年4月16日
    130
  • 【15】进大厂必须掌握的面试题-容器化和虚拟化面试

    Q1。什么是容器? 我的建议是首先解释对容器化的需求,容器用于提供一致的计算环境,从开发人员的笔记本电脑到测试环境,从过渡环境到生产环境。 现在给出容器的定义,一个容器包含一个完整…

    2020年10月23日
    414
  • 大数据在应急管理中的应用[通俗易懂]

    大数据在应急管理中的应用[通俗易懂]随着互联网、社交媒体和人工智能的技术发展和应用普及,大数据在应急管理中发挥的作用将越来越重要,是应急管理未来发展的重要方向之一。应急管理部的成立为中国应急管理的发展提供了政策上的支持,也为发展大数据在中国应急管理中的应用提供了契机。现阶段,理论研究尚无法完全预知大数据在应急管理中的具体应用。但基于对应急管理基本原理的掌握,结合对大数据本质属性的理解和对中国应急管理制度情境的了解,我们可以初步厘清大…

    2022年5月8日
    88
  • 【ArcGIS Pro微课1000例】0016:ArcGIS Pro 2.8浮雕效果地图制图案例教程[通俗易懂]

    【ArcGIS Pro微课1000例】0016:ArcGIS Pro 2.8浮雕效果地图制图案例教程[通俗易懂]ArcGISPro制作地图时可以制作出很多很炫的效果,比如地图阴影、地图晕渲效果、浮雕效果、三维效果等等。本实验讲解在ArcGISPro2.8中制作浮雕效果地图,效果如下所示:【参考阅读】:ArcGIS实验教程——实验四十四:ArcGIS地图浮雕效果制作完整案例教程1.加载矢量数据加载实验数据包data16.rar中的秦安县乡镇矢量数据:2.缓冲区分析点击【分析】选项卡,点击【缓冲区】。输入要素选择秦安县乡镇数据,选择输出要素路径,线性单位输入-0.4,单位为千米,侧类型选择.

    2022年9月5日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号