基于查表的整数霍夫变换方法实现(matlab)
暂时先用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)的更多相关文章
- 用MATLB仿真一个单闭环控制量,同时还存在两个开环控制变量的阶跃响应曲线。(自动控制方法是PID中的P控制。通过查表法直接给开环参数稳态最佳的大小)
实际项目背景:甘肃省,航天510所的LIPS100电推力器.一共有三个控制变量,开环控制变量是:Ia(阳极电流).mmrf(阳极主流率) 这个阳极主流率是阀门变量,不能够突变,模拟用(大学一年级课,电 ...
- python-面向对象速查表-内置方法-内置函数-内置属性(只整理了部分内容)
今日临时总结的内容,可能还有些不正确的地方,初步当做个速查表吧. 类的内置函数(继承object的,自己重写) 内置函数 执行时机 注意点 调用案例 __init__ 实例化对象时 不允许写返回值(r ...
- 基于跳跃表的 ConcurrentSkipListMap 内部实现(Java 8)
我们知道 HashMap 是一种键值对形式的数据存储容器,但是它有一个缺点是,元素内部无序.由于它内部根据键的 hash 值取模表容量来得到元素的存储位置,所以整体上说 HashMap 是无序的一种容 ...
- 表单数据验证方法(二)——ASP.NET后台验证
昨天写了一下关于如何在前台快捷实现表单数据验证的方法,今天接着昨天的,把后台实现数据验证的方法记录一下.先说明一下哈,我用的是asp.net,所以后台验证方法也是基于.net mvc来做的. 好了,闲 ...
- 基于稀疏表(Sparse Table)的RMQ(区间最值问题)
在RMQ的其他实现方法中,有一种叫做ST的算法比较常见. [构建] dp[i][j]表示的是从i起连续的2j个数xi,xi+1,xi+2,...xi+2j-1( 区间为[i,i+2j-1] )的最值. ...
- GNU Emacs命令速查表
GNU Emacs命令速查表 第一章 Emacs的基本概念 表1-1:Emacs编辑器的主模式 模式 功能 基本模式(fundamental mode) 默认模式,无特殊行为 文本模式(text m ...
- cmake 常用变量和常用环境变量查表手册
cmake 常用变量和常用环境变量查表手册 一,cmake 变量引用的方式: 前面我们已经提到了,使用${}进行变量的引用.在 IF 等语句中,是直接使用变量名而不通过${}取值 二,cmake 自定 ...
- 也来写写基于单表的Orm(使用Dapper)
前言 这两天看园子里有个朋友写Dapper的拓展,想到自己之前也尝试用过,但不顺手,曾写过几个方法来完成自动的Insert操作.而对于Update.Delete.Select等,我一直对Diction ...
- 华为基于策略划分VLAN的配置方法及示例
学过思科交换机的朋友,可能对基于策略划分VLAN的配置方法印象非常深,感觉确实比较复杂,先要配置VMPS以及VMPS数据库,但在华为交换机中,这种现象得到了彻底改变,因为它有了一种特殊的端口类型—— ...
随机推荐
- FarPoint.Win.Spread 自定义表头
最近C/S项目中用到FarPoint.Win.Spread,想在表头加个全选的checkbox,实现效果如图: 列的设置大家都清楚,直接可视化视图中设置该列CellType为CheckBox类型即 ...
- 京东笔试---通过考试(DP)
题目描述 小明同学要参加一场考试,考试一共有n道题目,小明必须作对至少60%的题目才能通过考试.考试结束后,小明估算出每题作对的概率,p1,p2,...,pn,你能帮他算出他通过考试的概率吗 ...
- java写文件读写操作(IO流,字符流)
package copyfile; import java.io.*; public class copy { public static void main(String[] args) throw ...
- java写文件读写操作(IO流,字节流)
package copyfile; import java.io.*; public class copy { public static void main(String[] args) throw ...
- Python CGI编程(转自易百)
Python CGI编程 Python的CGI编程,公共网关接口或CGI,Web服务器和一个自定义的脚本之间交换信息是一组定义的标准. 什么是CGI ? 公共网关接口或CGI,Web服务器和一 ...
- 查询表达式和LINQ to Objects
查询表达式实际上是由编译器“预处理”为“普通”的C#代码,接着以完全普通的方式进行编译.这种巧妙的发式将查询集合到了语言中,而无须把语义改得乱七八糟 LINQ的介绍 LINQ中的基础概念 降低两种数据 ...
- 蓝桥杯-趣味算式-java
/* (程序头部注释开始) * 程序的版权和版本声明部分 * Copyright (c) 2016, 广州科技贸易职业学院信息工程系学生 * All rights reserved. * 文件名称: ...
- httpd配置ResponseHeader
今天遇到一个问题:我把项目编译后的静态文件发布到开发机上,开发机使用httpd启的静态文件服务,页面的访问是在特制的壳浏览器里面,我更新了代码后,发现页面被缓存了,找到壳的RD联调了一下,发现我的主页 ...
- Oracle清除数据库中长时间占用资源的非活动的会话
1.启动资源计划 alter system set resource_limit=true scope=spfile; 2.设置非活动回话十五分钟断开,释放资源 alter profile defau ...
- Linux下Oracle的启动和关闭
默认情况下,Linux下Oracle是不会随系统自动启动的. 1.启动Oracle 1.以oracle账户登录到CentOS,或者切换到oracle用户权限 # su – oracle 2.然后输入 ...