For a square pixel grid, the ideal 2-d low-pass filter with a horizontal and a vertical cut-off angular frequency ωc in radians has an impulse response (kernel) h◻(x,y) that is the product of a horizontal and a vertical stretched and scaled sinc function, with x and y the integer horizontal and vertical pixel coordinates:
h◻[x,y]=ωcsinc(ωcxπ)π×ωcsinc(ωcyπ)π={ω2cπ2if x=y=0,sin(ωcx)sin(ωcy)π2xyotherwise.
If ωc=π, the kernel of Eq. 1 is simply:
h◻=[1]ifωc=π.
For real-valued x and y, Eq. 1 would not describe a circularly symmetric kernel, and neither is the square-like frequency response H◻ circularly symmetric. Sometimes an isotropic kernel may be desired, for example if we want to remove the effect of the choice of the direction of the image axes.
What is a kernel h∘[x,y] such that the frequency response H∘ is 1 inside a circle of radius ωc (angular frequency in radians) and 0 outside it?
Answer
Excerpted from Jae S.Lim 2D signal and image processing ch.1, as an example of 2-D circularly symmetric lowpass filter with a cutoff frequency of ωc radians per sample, whose impulse response is given by: h[n1,n2]=ωc2π√n21+n22J1(ωc√n21+n22)
where J1 is the Bessel function of the first kind and the first order...
Interested readers may consult the book for a derivation which is not untitive but nevertheless tractable; familiarity with Bessel functions is required but also provided as is.[Derivation is added.]
[Olli below]
At n1=n2=0 the limiting value must be used:
h[0,0]=ω2c4π
A slice through the middle of h[n1,n2] with ωc=π:
Python source for 2-d filter kernel (you may want to apply a 2-d window function):
from scipy import special
import numpy as np
def circularLowpassKernel(omega_c, N): # omega = cutoff frequency in radians (pi is max), N = horizontal size of the kernel, also its vertical size, must be odd.
kernel = np.fromfunction(lambda x, y: omega_c*special.j1(omega_c*np.sqrt((x - (N - 1)/2)**2 + (y - (N - 1)/2)**2))/(2*np.pi*np.sqrt((x - (N - 1)/2)**2 + (y - (N - 1)/2)**2)), [N, N])
kernel[(N - 1)//2, (N - 1)//2] = omega_c**2/(4*np.pi)
return kernel
Example with ωc=π:
import matplotlib.pyplot as plt
kernelN = 11 # Horizontal size of the kernel, also its vertical size. Must be odd.
omega_c = np.pi # Cutoff frequency in radians <= pi
kernel = circularLowpassKernel(omega_c, kernelN)
plt.imshow(kernel, vmin=-1, vmax=1, cmap='bwr')
plt.colorbar()
plt.show()
Example with ωc=π/4:
kernelN = 41 # Horizontal size of the kernel, also its vertical size. Must be odd.
omega_c = np.pi/4 # Cutoff frequency in radians <= pi
kernel = circularLowpassKernel(omega_c, kernelN)
plt.imshow(kernel, vmin=-np.max(kernel), vmax=np.max(kernel), cmap='bwr')
plt.colorbar()
plt.show()
[fat32 below]
Ok, for those who would benefit from a (not so intuitive) derivation, here I make a (n almost) verbatim copy of it from the same book.
First, lets simply write the 2D discrete-time inverse Fourier transform to define the impulse response as:
h[n1,n2]=1(2π)2∫∫ω21+ω22<w2c1⋅ej(ω1n1+ω2n2)dω1dω2
Let's make a change of variables ω1=rcos(θ) and ω2=rsin(θ) (effectively write the integral in (1) in the cyclindrical (or polar) coordinates) :
h[n1,n2]=1(2π)2∫ωcr=0∫a+2πθ=aejr(cos(θ)n1+sin(θ)n2)r drdθ
Now, make a further change of variables n1=ncos(ϕ) and n2=nsin(ϕ), with n=√n21+n22 and obtain
h[n1,n2]=1(2π)2∫ωcr=0r dr∫a+2πθ=aejrn(cos(θ)cos(ϕ)+sin(θ)sin(ϕ))dθ
which is now: h[n1,n2]=1(2π)2∫ωcr=0r dr∫a+2πθ=aejrncos(θ−ϕ)dθ
defining f(r)=∫a+2πθ=aejrncos(θ−ϕ)dθ , then we get:
h[n1,n2]=1(2π)2∫ωcr=0rf(r)dr
Now, exploring f(r) we can apply the Euler's identity :
f(r)=∫a+2πθ=acos(r.n.cos(θ−ϕ))dθ+j∫a+2πθ=asin(r.n.cos(θ−ϕ))dθ
And we can notice that the imaginary part is zero (can check an integral table for that) and with a=ϕ then f(r) becomes
f(r)=∫2πθ=0cos(r.n.cos(θ))dθ
Now the integral in (7) is recognized as the the Bessel function of order zero, kind one J0(x) which is given as:
J0(x)=12π∫2πθ=0cos(xcos(θ))dθ
from (7) and (8) we see that f(r)=2πJ0(rn)...
And the last identity is given as is: xJ1(x)|ba=∫baxJ0(x)dx
Now putting the impulse response into the form
h[n1,n2]=1(2π)2∫ωcr=0r2πJ0(rn)dr
applying (9) onto (10) with the recognition that x=rn and dr=dx/n and n=√n21+n22 yields the result:
h[n1,n2]=ωc2π√n21+n22J1(ωc√n21+n22)
No comments:
Post a Comment