|
|
EDA365欢迎您登录!
您需要 登录 才可以下载或查看,没有帐号?注册
x
1 A0 H: |8 U! |0 N+ m. G霍夫变换Hough0 n; o+ q4 Z$ r- `* ~
霍夫变换(Hough)是一个非常重要的检测间断点边界形状的方法。它通过将图像坐标空间变换到参数空间,来实现直线与曲线的拟合。6 t& K+ K) P- H
& [ Y4 P7 ], T1.直线检测% G; B# R9 P( o% C- z' B
; S" C1 \, j2 r% H1.1 直线坐标参数空间9 {9 m2 \6 F+ K5 B7 c+ Y* w
/ X8 N* b$ y9 X5 N8 G在图像x−y坐标空间中,经过点(xi,yi)的直线表示为:
4 e& ~/ g( J8 p/ R( O( {" W# G5 D- S9 N0 F* C
yi=axi+b(1)
$ k4 U9 J5 K/ c& Z9 P: Y4 m
- |0 G4 r: Y* s* R其中,参数a为斜率,b为截矩。 1 f) D3 H, q$ Q7 I: g& M6 r
/ n9 s4 ` `7 I3 Z# i. {" N通过点(xi,yi)的直线有无数条,且对应于不同的a和b值。
* ^" s( t5 j/ E8 M$ }, B9 W$ b, [( m9 N- ]: X
如果将xi和yi视为常数,而将原本的参数a和b看作变量,则式子(1)可以表示为:: C" J9 G* P# Q% @ N# h
3 I* C2 f* m& x/ e/ J$ F
b=−xia+yi(2)
" D: | H, [2 o, d7 J/ U' a6 a
2 h B* k) j0 R; p3 P/ [这样就变换到了参数平面a−b。这个变换就是直角坐标中对于(xi,yi)点的Hough变换。
( s+ _$ k. `6 X# m/ P% @: Y7 @; g* w0 Q, y3 n& q
该直线是图像坐标空间中的点(xi,yi)在参数空间的唯一方程。考虑到图像坐标空间中的另一袋奶(xj,yj),它在参数空间中也有相应的一条直线,表示为:5 p; K6 n3 U- M& ?6 J
R) c8 {4 x# r* i: x, N9 Tb=−xja+yj(3)
) i, U* [; k6 Q, f( S
9 I) |7 e" d. o2 d% n这条直线与点(xi,yi)在参数空间的直线相交于一点(a0,b0),如图所示:
6 l$ c# ^9 D9 t; n0 _- u$ M& g9 Z
+ }) N6 m' k6 v) e. z
1 N6 s9 e& _3 U
9 ^; H( {) }' Q1 ]
图像坐标空间中过点(xi,yi)和点(xj,yj)的直线上的每一点在参数空间a−b上各自对应一条直线,这些直线都相交于点(a0,b0),而a0、b0就是图像坐标空间x−y中点(xi,yi)和点(xj,yj)所确定的直线的参数。 : A; W I8 ?5 H7 O; w" I8 l
6 u6 u0 j" H5 b: U5 Q; M& ?2 U
反之,在参数空间相交于同一点的所有直线,在图像坐标空间都有共线的点与之对应。根据这个特性,给定图像坐标空间的一些边缘点,就可以通过Hough变换确定连接这些点的直线方程。
: \9 Q$ ~5 T6 D! @; _2 p: p* q9 I4 J9 ]8 u
4 e% _! H6 O0 u2 p& i; S5 C8 F
具体计算时,可以将参数空间视为离散的。建立一个二维累加数组A(a,b),第一维的范围是图像坐标空间中直线斜率的可能范围,第二维的范围是图像坐标空间中直线截矩的可能范围。开始时A(a,b)初始化为0,然后对图像坐标空间的每一个前景点(xi,yi),将参数空间中每一个a的离散值代入式子(2)中,从而计算出对应的b值。每计算出一对(a,b),都将对应的数组元素A(a,b)加1,即A(a,b)=A(a,b)+1。所有的计算结束之后,在参数计算表决结果中找到A(a,b)的最大峰值,所对应的a0、b0就是源图像中共线点数目最多(共A(a,b)个共线点)的直线方程的参数;接下来可以继续寻找次峰值和第3峰值和第4峰值等等,它们对应于原图中共线点略少一些的直线。
( S7 w4 y1 ^2 E/ E5 \' }5 z& K! A1 v, B
注意:由于原图中的直线往往具有一定的宽度,实际上相当于多条参数极其接近的单像素宽直线,往往对应于参数空间中相邻的多个累加器。因此每找到一个当前最大的峰值点后,需要将该点及其附近点清零,以防算法检测出多条极其邻近的“假”直线。9 `0 D) n- d. m( _8 t' |; [1 q
% Y; j8 p2 \! [, }/ N
$ z2 D4 i8 p$ V2 e
对于上图的Hough变换空间情况如下图所示。 # D) P+ n) f! g) f
+ G4 L/ r. W7 l# G& r% w
5 ~5 i# e7 ?$ k+ [, n& w
& m; P) p2 H& s) ]这种利用二维累加器的离散方法大大简化了Hough变换的计算,参数空间a−b上的细分程度决定了最终找到直线上点的共线精度。上述的二维累加数组A也被称为Hough矩阵。# B# H7 P; l: B3 h
1 b9 J/ G+ t0 A8 a! W1 Y/ ~注意:使用直角坐标表示直线,当直线为一条垂直直线或者接近垂直直线时,该直线的斜率为无限大或者接近无限大,从而无法在参数空间$a-b$上表示出来。为了解决这个问题,可以采用极坐标。4 K; ]! G2 T- x( S( z
0 G2 d! S9 K" ?2 k5 l
. f2 p3 ?' S) f; o1.2 极坐标参数空间$ M7 |# }! n0 h; B M3 a7 s
( L& |0 ~7 L* l& G极坐标中用如下参数方程表示一条直线。3 n9 t3 q% K$ V# c
0 B% r4 @; J: e: Y/ ]4 g, p) ?8 E
ρ=xcosθ+ysinθ(4)0 q4 ]+ } S# { w/ s. F
4 D$ |) Z8 ~ f& ]* n0 d其中,ρ代表直线到原点的垂直距离,θ代表x轴到直线垂线的角度,取值范围为±90°,如图所示。
& ?* j% Z. a3 C7 }
: | ^7 t8 }1 H$ N
3 g0 _* U" s( Y' O5 ]与直角坐标类似,极坐标中的Hough变换也将图像坐标空间中的点变换到参数空间中。 % ~0 x. q9 }& N: n2 c
在极坐标表示下,图像坐标空间中共线的点变换到参数空间中后,在参数空间都相交于同一点,此时所得到的ρ、θ即为所求的直线的极坐标参数。与直角坐标不同的是,用极坐标表示时,图像坐标空间的共线的两点(xi,yi)和(xj,yj)映射到参数空间是两条正弦曲线,相交于点(ρ0,θ0),如上图所示。, \9 k; G% g* [& p u
; A" G; f; g8 W- h* t6 \
具体计算时,与直角坐标类似,也要在参数空间中建立一个二维数组累加器A,只是取值范围不同。对于一副大小为D×D的图像,通常ρ的取值范围为[−2–√D/2,2–√D/2],θ的取值范围为[−90°,90°]。计算方法与直角坐标系中累加器的计算方法相同,最后得到最大的A所对应的(ρ,θ)。
) o# ^' k( L- I" e3 O9 q
9 H/ n7 X6 M* Q4 L3 a2.曲线检测, _6 P1 c; {7 j
b) I8 p0 o. ~, i
Hough变换同样适用于方程已知的曲线检测。8 | Z. q ~! t( M) m0 H
8 G. `. @8 j# I2 D! q3 o
图像坐标空间中的一条已知的曲线方程也可以建立其相应的参数空间。由此,图像坐标空间中的一点,在参数空间中就可以映射为相应的轨迹曲线或者曲面。
. w, m9 h: f& _) d' k
, ?8 S' e, q7 W# v3 k# P/ W% O" L1 r8 G若参数空间中对应各个间断点的曲线或者曲面能够相交,就能找到参数空间的极大值以及对应的参数;若参数空间中对应各个间断点的曲线或者曲面不能相交,则说明间断点不符合某已知曲线。0 r! s: X9 t7 B
- a% q; Z: ?& N
Hough变换做曲线检测时,最重要的是写出图像坐标空间到参数空间的变换公式。
( u' P+ v: J3 d% @' ^5 G: R
9 _) I. S3 v. j0 f, |' u! @- E' }例如,对于已知的圆方程,其直角坐标的一般方程为: , C; R$ D# V7 t1 k" w# ]# O/ [
5 n0 _ w0 Y, q
(x−a)2+(y−b)2=r2(5)
: A5 M6 g( r8 y$ _. J
+ g/ d2 j' D: @' A8 \$ |2 w其中,(a,b)为圆心坐标,r为圆的半径。
. U! L' }0 G; `! v7 V/ @' J) A$ L! ?8 q8 A" l% }
那么,参数空间可以表示为(a,b,r),图像坐标空间中的一个圆对应参数空间中的一个点。
+ q& l- y; X9 u; @, P4 I, n* ~" `8 @; J3 I, d
具体计算时,与前面讨论的方法相同,只是数组累加器为三维A(a,b,r)。
2 D& F( m; \' v' P8 B! [8 e
3 Z+ C1 V- d5 Z# o+ W计算过程是让a,b在取值范围内增加,解出满足上式的r值,每计算出一个(a,b,r)值,就对相应的数组元素A(a,b,r)加1.计算结束后,找到的最大的A(a,b,r)所对应的a,b,r就是所求的圆的参数。: D4 `7 O$ v4 c# S. H4 `, \
5 y! y$ q/ q; \, c, |$ f% ~' @
与直线检测一样,曲线检测也可以通过极坐标形式计算。 * h t; x4 S* ^$ l2 z' Z
% Q- L; w; ~* J注意:通过Hough变换做曲线检测,参数空间的大小将随着参数个数的增加呈指数增长的趋势。所以在实际使用时,要尽量减少描述曲线的参数数目。因此,这种曲线检测的方法只对检测参数较少的曲线有意义。1 R1 F/ k& g; k* j7 T* u
. ?: ^& }% l' m: z7 N8 w3 x0 Q: T# Y8 d' H
3.任意形状的检测
% F! Y, \) ^: A& T$ z8 U8 |% c/ L8 ~. z; d0 J3 b
这里所说的任意形状的检测,是指应用广义Hough变换去检测某一任意形状边界的图形。它首先选取该形状中的任意点(a,b)为参考点,然后从该任意形状图形的边缘每一点上,计算其切线方向ϕ和到参考点(a,b)位置的偏移矢量r,以及r与x轴的夹角α。4 |2 ?8 ~; P9 {2 H" n# ?: [
! y: E0 n: r. { y$ a
参考点(a,b)的位置可由下式算出:
0 h% A9 o8 ?7 O9 q0 @( ^
1 B- u8 r$ T( }5 Xa=x+r(ϕ)cos(α(ϕ))(6)
8 N7 _; Y8 b1 m. w& L- ~ |
# N7 F% `( D+ M# H ]% a: hb=x+r(ϕ)sin(α(ϕ))(7)3 q& M5 X% Z2 {0 d
8 {8 J0 v' m+ O" w2 B9 B; n" x* Y- @# n7 M6 d( d
2.Hough变换直线检测的Matlab实现0 H& h! n O K8 n/ m
; J0 w* g7 r" Z% B通过Hough在二值图像中检测直线需要以下3个步骤。
1 k Q( y' z9 Z% K$ _/ E, r) r* ~·>(1)利用hough()函数执行霍夫变换,得到霍夫矩阵。
1 S( w$ q9 w1 r. F* \: s·>(2)利用houghpeaks()函数在霍夫矩阵中寻找峰值点。 " P ^4 h' ~9 G6 i0 N3 P. d* k% C$ n
·>(3)利用houghlines()函数在之前2步结果的基础上得到原二值图像中的直线信息。/ Q5 p' I+ P- s" @# |6 [8 y" u
/ G8 X8 g# ^1 {5 N# T/ R: e0 [2.1 霍夫变换–Hough' _ m1 C; \5 z6 K+ h, F
0 h4 G0 x6 v3 j9 g9 K2 w: x
调用形式:
8 z5 i8 G; Q& [, z U2 U% G% \3 ]! l) U ^% v
[H,theta,rho]=hough(BW,param1,value1,param2,value2)
- ]1 ^ |8 {, Y) u( C5 N
/ ^6 t& b. l' N2 \3 v3 e: f0 [# G参数说明: 7 {" c) a' E% \5 a# D
+ T. I. g$ S: ?& E$ P6 y, K·BW是边缘检测后的二值图像; % F9 W" L/ n |% Y2 ]; z9 ~5 W
·可选参数对param1,value1和param2,value2的合法取值如下:
) k; j; x3 V/ ~3 B* u1 F! _' k" W2 g, h. @( ^6 j- b
param取值 含义
7 w1 T" K6 b1 ~( dThetaResolution Hough矩阵中θ轴方向上单位区间的长度(以“度”为单位),可取(0,90)区间上的实数,默认为11 _6 B) C: G7 B2 F1 V; M; ~
RhoResolution Hough矩阵中\rho轴方向上单位区间的长度,可取(0,norm(size(BW)))区间上的实数,默认为1
$ s" s# t( O6 b$ A/ E' ?1 t2 h
8 Z) O7 `) i& K7 H* k- N5 t( |% r) x返回值:
# b% \2 K9 Z! F& ]/ u' T·H是变换得到的霍夫矩阵
/ g3 _# `0 `3 z/ H·theta,rho分别对应于Hough矩阵每一列和每一行的\theta和\rho值组成的向量。/ B3 Q$ U; M/ U# \/ Y& P* u
: C8 m+ c: e/ E9 z- 补充:[m,n] = size(X)
- 补充:norm的用法,matlab help norm
- NORM Matrix or vector norm.
- For matrices...
- 对于矩阵...
- NORM(X) is the largest singular value of X, max(svd(X)).
- NORM(X)是X的最大奇异值
- NORM(X,2) is the same as NORM(X).
- NORM(X,1) is the 1-norm of X, the largest column sum,
- = max(sum(abs(X))).
- NORM(X,inf) is the infinity norm of X, the largest row sum,
- = max(sum(abs(X'))).
- NORM(X,'fro') is the Frobenius norm, sqrt(sum(diag(X'*X))).
- NORM(X,P) is available for matrix X only if P is 1, 2, inf or 'fro'.
- For vectors...
- 对于向量...
- NORM(V,P) = sum(abs(V).^P)^(1/P).
- 返回向量A的p范数
- NORM(V) = norm(V,2).
- 返回向量A的2范数,即欧几里德范数。二范数等价于平方和开平方,Sqrt(X1^2+X2^2+...+Xn^2)
8 ^! l3 m5 h6 C a. j
" x3 g: k) N) Y5 i O: `2 R# f
% \6 M0 V- B: x2.2 寻找峰值–houghpeaks
# p$ M( _/ _( [6 p
7 w- b& ~4 E0 {$ r8 ^+ `0 N5 C调用形式: : x" u1 o! Z6 A
. |9 U) b1 h. {peaks=houghpeaks(H,numpeaks,param1,value1,param2,value2)0 ^/ y+ x% h8 u" r
# w1 U2 b5 H( n5 s1 C8 ?1 K. m+ }8 K
参数说明: 4 d- @9 o; g7 n0 k9 B/ s% K
·H是hough()函数得到的霍夫矩阵 ; \# J0 |0 X0 b
·numpeaks是要寻找的峰值数目,默认为1
1 x5 L% T0 ^1 f/ j- V" z·可选参数对param1,value1和param2,value2的合法取值如下:
* {7 [$ W3 {5 P
% G" h: o% C! k# [param取值 含义
; J% x/ E7 K( DThreshold 峰值的阈值,只有大于该阈值的店才被认为是可能的阈值,取值>0,默认为0.5×max(H(:)): M6 ^ o8 @8 Q8 D7 m
NHoodSize 在每次检测出一个峰值后,NhoodSize指出了在该峰值周围需要清零的领域信息。以[m,n]的形式给出,其中m、n均为正的奇数。默认为大于等于size(H)/50的最小奇数。8 W8 ?+ k# @ b6 _& L X
& p+ {& F L5 ?. p( Q& b
" J" y2 x/ E. r# U+ \" }
返回值: & ^ P$ c* `% z
·peaks是一个Q×2的矩阵,每行的两个元素分别是某一峰值点再hough矩阵中的行、列索引,Q为找到的峰值点的数目。
1 u9 _ X K! i2 x! P7 }8 L$ s- Q
. v2 R7 g4 q: c3 J' P$ ]
2.3 提取直线段–houghlines; X z" t5 V6 _% \7 P3 e9 w
+ P. ?6 p, R# K T% k" @调用形式:
- @; A' u7 q* l/ W5 ~, D) @# B3 z# W4 C/ r0 U8 V5 J6 E% c
lines=houghlines(BW,theta,rho,peaks,param1,value1,param2,value2)5 ^; a% E; Z" D7 R+ a
" A3 ~7 c; E3 A4 E. `* a& M参数说明:
; n5 A! `4 ?' |' A# d; }* T·BW是边缘检测后的二值图像
0 T; u, {% `8 S8 l; T) T·theta,rho分别对应于Hough矩阵每一列和每一行的θ和ρ值组成的向量。有hough()函数返回。
8 K4 q2 G9 l3 Z" Q9 e·peaks是一个包含峰值点信息的Q×2的矩阵,由houghpeaks()函数返回。 # a K& _0 w; h4 u7 l
·可选参数对param1,value1和param2,value2的合法取值如下:
, o6 |8 v1 H/ a& e& H$ v) u" ^! ?; }' r4 I6 t7 M+ `
param取值 含义
: U$ |/ B& t _, p9 m3 L/ SFillGap 线段合并的阈值:如果对应于Hough矩阵某一个单元格(相同的θ和ρ)的两条线段之间的距离小于 FillGap,则合并为一个直线段。默认值为20.
/ Q! E" C' \! @- J7 `7 A( s7 tMinLength 检测的直线段的最小长度阈值:如果检测出的直线段长度大于MinLength,则保留;丢弃小于MinLength的直线段。默认值为40.6 T, u, H' ^0 b
! C1 G/ o! G$ u z* M6 u0 i# g' K
. N9 {; g- Q7 }4 o$ Y0 K% j2 n8 {返回值: 9 ~4 P/ D! b3 o5 ]$ j
·lines是一个结构体数组,数组长度是找到的直线条数,而每一个数组元素(直线段结构体)的内部结构如下:
+ g3 u& b# D3 K
; h u) h; V) @! G; Y S: q% R2 p' q域 含义) J' o# ?0 O, W+ T
point1 直线段的端点1/ o, J3 x+ M+ J
point2 直线段的端点2# i5 y# n1 J9 l9 a! ?; y. V
thea 对应在霍夫矩阵中的θ
0 a7 H4 |5 U2 P( trho 对应在霍夫矩阵中的ρ8 [- B) ]; M) `; T
& _6 N5 [% w& g- D2 D
8 W; y* z4 ?9 S6 d3 k- c2.4 Hough变换的matlab实现
6 r6 |( |( |- t' S3 b; L1 Z1 q0 c% s: V8 h$ A. e+ a) o* `& X
matlab中输入帮助命令:
# v# ^1 f6 i! _& z1 G/ ^help hough; help houghpeaks; help houghlines ) I' \3 W2 \ n
就可以看到代码示例,很详细,其实用好帮助文档一切问题都可以解决。
6 }9 N" W' u" [) {& x
% v4 T8 `" Y9 c+ T: uExample% g5 ]* I7 x( }$ b, p
; q5 s, i0 d* q& ISearch for line segments corresponding to five peaks in the Hough
* H X9 ?" B: O- A2 a* D) e+ ntransform of the rotated circuit.tif image. Additionally, highlight
; G J5 e1 P ^- c$ }- k! ?& pthe longest segment./ m" b) _6 z, ]8 B& {: v
) e- X3 C3 J0 j2 l% E! |' }
- I = imread('circuit.tif');
- rotI = imrotate(I,33,'crop');
- BW = edge(rotI,'canny');
- [H,T,R] = hough(BW);
- imshow(H,[],'XData',T,'YData',R,'InitialMagnification','fit');
- xlabel('\theta'), ylabel('\rho');
- axis on, axis normal, hold on;
- P = houghpeaks(H,5,'threshold',ceil(0.3*max(H(:))));
- x = T(P(:,2));
- y = R(P(:,1));
- plot(x,y,'s','color','white');
- % Find lines and plot them
- lines = houghlines(BW,T,R,P,'FillGap',5,'MinLength',7);
- figure, imshow(rotI), hold on
- max_len = 0;
- for k = 1:length(lines)
- xy = [lines(k).point1; lines(k).point2];
- plot(xy(:,1),xy(:,2),'LineWidth',2,'Color','green');
- % plot beginnings and ends of lines
- plot(xy(1,1),xy(1,2),'x','LineWidth',2,'Color','yellow');
- plot(xy(2,1),xy(2,2),'x','LineWidth',2,'Color','red');
- % determine the endpoints of the longest line segment
- len = norm(lines(k).point1 - lines(k).point2);
- if ( len > max_len)
- max_len = len;
- xy_long = xy;
- end
- end
- % highlight the longest line segment
- plot(xy_long(:,1),xy_long(:,2),'LineWidth',2,'Color','cyan');
) N6 o A* k5 j6 u ( _& z) t$ G% ^; e0 [
+ |. L! L* a2 B6 p
( J" {3 L+ Z! S: ^2 ~: O |
|