暂时先用matlab把算法弄一下,这是基于查表的整数霍夫变换方法实现及解释。

接着再实现FPGA的霍夫变换。

霍夫变换原理和算法这里不多说,可参考以下链接:

http://blog.csdn.net/poem_qianmo/article/details/26977557/

http://www.ilovematlab.cn/thread-25436-1-1.html

论文:基于FPGA的实时整数霍夫变换_唐林波

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

霍夫变换分以下几个步骤:

Step1:输入640*480图像,并进行边缘检测(建议用sobel,因为FPGA较容易实现)

Step2:建立(ρ,θ)坐标系的霍夫矩阵,其中rho_max为图像对角线长度sqrt(640^2+480^2)=800

Step3 建立cos,sin值查表θ=[0 180],其中cos,sin值乘以1000倍取整。

Step4 计算所有非零像素点的ρ,θ坐标并累加

Step5 寻找霍夫矩阵中前N个最大值

Step6 显示霍夫变换矩阵及显示前N个最大值点

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

坐标变换推导:

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

具体matlab代码如下:

clc;
clear all
close all
%Step %%提取边缘检测图像%%%%%%%
RGB = imread('112.jpg');%*
I=rgb2gray(RGB); % 图片用的是灰度图像
[x,y]=size(I);
BW=edge(I,'sobel');
% figure();imshow(I);title('原图')
% figure();imshow(BW);title('边缘检测图像') %Step2 %%建立ρ,θ坐标系矩阵%%%%%%%%
rho_max = ; %%sqrt(^+^)
accarray=zeros(rho_max,); %定义ρ,θ坐标系的数组,初值为0。
%θ的最大值,180度
%Step3 建立cos,sin值表 θ=[ ]
Theta=[:pi/:pi/]; %定义θ数组,确定θ取值范围,细分180份 °
cosvalue=zeros(,);%%新建cos值表
sinvalue=zeros(,);%%新建sin值表
for k=: %%计算0-90度角度值
cosvalue(,k) = floor(cos(Theta(k))*);
sinvalue(,k) = floor(sin(Theta(k))*);
end %Step4 计算所有非零像素点的ρ,θ坐标并累加
for n=:x,
for m=:y
if BW(n,m)== %%如果像素是1 有边缘
for k=:
%将θ值代入hough变换方程,求ρ值
if(k<=)
rho=(m*cosvalue(,k))+(n*sinvalue(,k)); %%-°
else
rho=(n*cosvalue(,k-))-(m*sinvalue(,k-)); %%-°
end
%将ρ值与ρ最大值的和的一半作为ρ的坐标值(数组坐标),这样做是为了防止ρ值出现负数
rho_int=round(rho/+rho_max/);
%在ρθ坐标(数组)中标识点,即计数累加
accarray(rho_int,k)=accarray(rho_int,k)+;
end
end
end
end
%%Step5 寻找霍夫矩阵中前N个最大值
N = ;
b=sort(reshape(accarray,,*));
[m,n]=find(accarray>b(*-N));
TT=[m,n]; %%Step6 显示霍夫变换矩阵及显示前N个最大值点
figure();
colormap gray;
imagesc(accarray);title('hough变换后的图')
set(gca,'YTick',::);
set(gca,'XTick',::);
xlabel('\theta'), ylabel('\rho');
hold on
plot(n,m,'*r');

~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

结果分析:

输入图像:

图1

霍夫变换输出结果:

图2

图1与图2符号一一对应,可能你会发现两张图的ρ值不一致,是因为我们在Step4代码的计算中防止ρ值出现负数rho_int=round(rho/2000+rho_max/2);

相当于把±800的ρ值范围映射到[0 800]中,只需要进行简单换算就可以证实:

例A点: 560 = 320/2 + 400。

此外我们还可以观察到图1中有两个圆C1,和C2,在图二中标记出C1和C2的范围就是园内任意线段在霍夫空间中取值的范围,在C2外面仍有一个亮点H和G这是原图中边界线造成的。

由于FPGA内部不可能大量存储计算值,我们在设计霍夫变换算法的时候就应该尽可能考虑节省空间的方法。

今天先用matlab验证了基于查表的整数霍夫变换方法的可行性,当然还需要进一步优化,下一步将会改写成Verilog代码在FPGA上跑。

有兴趣的朋友欢迎交流,QQ:971789220

未完待续~~~  2017-6-6

基于查表的整数霍夫变换方法实现(matlab)的更多相关文章

  1. 用MATLB仿真一个单闭环控制量,同时还存在两个开环控制变量的阶跃响应曲线。(自动控制方法是PID中的P控制。通过查表法直接给开环参数稳态最佳的大小)

    实际项目背景:甘肃省,航天510所的LIPS100电推力器.一共有三个控制变量,开环控制变量是:Ia(阳极电流).mmrf(阳极主流率) 这个阳极主流率是阀门变量,不能够突变,模拟用(大学一年级课,电 ...

  2. python-面向对象速查表-内置方法-内置函数-内置属性(只整理了部分内容)

    今日临时总结的内容,可能还有些不正确的地方,初步当做个速查表吧. 类的内置函数(继承object的,自己重写) 内置函数 执行时机 注意点 调用案例 __init__ 实例化对象时 不允许写返回值(r ...

  3. 基于跳跃表的 ConcurrentSkipListMap 内部实现(Java 8)

    我们知道 HashMap 是一种键值对形式的数据存储容器,但是它有一个缺点是,元素内部无序.由于它内部根据键的 hash 值取模表容量来得到元素的存储位置,所以整体上说 HashMap 是无序的一种容 ...

  4. 表单数据验证方法(二)——ASP.NET后台验证

    昨天写了一下关于如何在前台快捷实现表单数据验证的方法,今天接着昨天的,把后台实现数据验证的方法记录一下.先说明一下哈,我用的是asp.net,所以后台验证方法也是基于.net mvc来做的. 好了,闲 ...

  5. 基于稀疏表(Sparse Table)的RMQ(区间最值问题)

    在RMQ的其他实现方法中,有一种叫做ST的算法比较常见. [构建] dp[i][j]表示的是从i起连续的2j个数xi,xi+1,xi+2,...xi+2j-1( 区间为[i,i+2j-1] )的最值. ...

  6. GNU Emacs命令速查表

    GNU Emacs命令速查表 第一章  Emacs的基本概念 表1-1:Emacs编辑器的主模式 模式 功能 基本模式(fundamental mode) 默认模式,无特殊行为 文本模式(text m ...

  7. cmake 常用变量和常用环境变量查表手册

    cmake 常用变量和常用环境变量查表手册 一,cmake 变量引用的方式: 前面我们已经提到了,使用${}进行变量的引用.在 IF 等语句中,是直接使用变量名而不通过${}取值 二,cmake 自定 ...

  8. 也来写写基于单表的Orm(使用Dapper)

    前言 这两天看园子里有个朋友写Dapper的拓展,想到自己之前也尝试用过,但不顺手,曾写过几个方法来完成自动的Insert操作.而对于Update.Delete.Select等,我一直对Diction ...

  9. 华为基于策略划分VLAN的配置方法及示例

     学过思科交换机的朋友,可能对基于策略划分VLAN的配置方法印象非常深,感觉确实比较复杂,先要配置VMPS以及VMPS数据库,但在华为交换机中,这种现象得到了彻底改变,因为它有了一种特殊的端口类型—— ...

随机推荐

  1. 【Java 并发】详解 ThreadPoolExecutor

    前言 线程池是并发中一项常用的优化方法,通过对线程复用,减少线程的创建,降低资源消耗,提高程序响应速度.在 Java 中我们一般通过 Exectuors 提供的工厂方法来创建线程池,但是线程池的最终实 ...

  2. spring学习总结一----控制反转与依赖注入

    spring作为java EE中使用最为广泛的框架,它的设计体现了很多设计模式中经典的原则和思想,所以,该框架的各种实现方法非常值得我们去研究,下面先对spring中最为重要的思想之一----控制反转 ...

  3. 根据GPS经纬度判断当前所属的市区

    这个事情分两步走 1. 拿到行政区划的地理围栏数据 2. 根据GPS定位判断一个点是否落在地理围栏的多边形区域里. 1. 获取行政区划的地理围栏数据可以利用百度API.打开以前我的一个例子在chrom ...

  4. AOJ/数据结构习题集

    ALDS1_3_A-Stack. Description: Write a program which reads an expression in the Reverse Polish notati ...

  5. bzoj1898 [Zjoi2005]沼泽鳄鱼

    Description 潘塔纳尔沼泽地号称世界上最大的一块湿地,它地位于巴西中部马托格罗索州的南部地区.每当雨季来临,这里碧波荡漾.生机盎然,引来不少游客.为了让游玩更有情趣,人们在池塘的中央建设了几 ...

  6. seajs模块标识命名和解析规则

    模块标识采用路径形式,但要注意与路径的区别.require.require.async的第一个参数是模块标识.而seajs.use第一个参数为文件路径. use是全局的,require是局部的.模块标 ...

  7. STL容器之优先队列(转)

    STL容器之优先队列 原地址:http://www.cnblogs.com/summerRQ/articles/2470130.html 优先级队列,以前刷题的时候用的比较熟,现在竟然我只能记得它的关 ...

  8. Python批量修改文件名与后缀

    引言: 有时因为文件版本的更新,后缀名会发生变化,例如Word13的docx到Word16的doc,又例如我们想修改音频文件的后缀.一个一个修改后缀名往往很麻烦,于是我们便可以写一个Python的脚本 ...

  9. AngularJS的相关应用

    一.[AngularJS常用指令]        1.ng-app:声明Angular所管辖的区域.一般写在body或html上,原则上一个页面只有一个:           <body ng- ...

  10. python-day 练习1

    #!/usr/bin/env python# -*- coding:utf-8 -*-'''需求: a. 元素分类 有如下值集合 v1 = [11,22,33,44,55,66,77,88,99,90 ...