mirror of
https://github.com/wassname/scikit-image.git
synced 2026-07-03 10:02:26 +08:00
122 lines
3.9 KiB
Cython
122 lines
3.9 KiB
Cython
"""template.py - Template matching
|
|
"""
|
|
import cython
|
|
cimport numpy as np
|
|
import numpy as np
|
|
from scipy.signal import fftconvolve
|
|
from skimage.transform import integral
|
|
|
|
|
|
cdef extern from "math.h":
|
|
float sqrt(float x)
|
|
float fabs(float x)
|
|
|
|
|
|
@cython.boundscheck(False)
|
|
cdef float sum_integral(np.ndarray[float, ndim=2, mode="c"] sat,
|
|
int r0, int c0, int r1, int c1):
|
|
"""
|
|
Using a summed area table / integral image, calculate the sum
|
|
over a given window.
|
|
|
|
This function is the same as the `integrate` function in
|
|
`skimage.transform.integrate`, but this Cython version significantly
|
|
speeds up the code.
|
|
|
|
Parameters
|
|
----------
|
|
sat : ndarray of float
|
|
Summed area table / integral image.
|
|
r0, c0 : int
|
|
Top-left corner of block to be summed.
|
|
r1, c1 : int
|
|
Bottom-right corner of block to be summed.
|
|
|
|
Returns
|
|
-------
|
|
S : int
|
|
Sum over the given window.
|
|
"""
|
|
cdef float S = 0
|
|
|
|
S += sat[r1, c1]
|
|
|
|
if (r0 - 1 >= 0) and (c0 - 1 >= 0):
|
|
S += sat[r0 - 1, c0 - 1]
|
|
|
|
if (r0 - 1 >= 0):
|
|
S -= sat[r0 - 1, c1]
|
|
|
|
if (c0 - 1 >= 0):
|
|
S -= sat[r1, c0 - 1]
|
|
return S
|
|
|
|
|
|
@cython.boundscheck(False)
|
|
def match_template(np.ndarray[float, ndim=2, mode="c"] image,
|
|
np.ndarray[float, ndim=2, mode="c"] template,
|
|
str method):
|
|
# convolve the image with template by frequency domain multiplication
|
|
cdef np.ndarray[float, ndim=2] result
|
|
# when `dtype=float` is used, ascontiguousarray returns ``double``.
|
|
result = np.ascontiguousarray(fftconvolve(image, np.fliplr(template),
|
|
mode="valid"), dtype=np.float32)
|
|
|
|
# calculate squared integral images used for normalization
|
|
cdef np.ndarray[float, ndim=2, mode="c"] integral_sum
|
|
cdef np.ndarray[float, ndim=2, mode="c"] integral_sqr
|
|
|
|
if method == 'norm-coeff':
|
|
integral_sum = integral.integral_image(image)
|
|
integral_sqr = integral.integral_image(image**2)
|
|
|
|
# use inversed area for accuracy
|
|
cdef float inv_area = 1.0 / (template.shape[0] * template.shape[1])
|
|
cdef float template_norm
|
|
cdef float template_mean = np.mean(template)
|
|
|
|
if method == 'norm-corr':
|
|
# calculate template norm according to the following:
|
|
# variance = 1/K Sum[(x_k - mean) ** 2]
|
|
# = 1/K Sum[x_k ** 2] - mean ** 2
|
|
#template_norm = sqrt((np.std(template) ** 2 +
|
|
#template_mean ** 2)) / sqrt(inv_area)
|
|
# TODO: check equation for template_norm.
|
|
# The above normalization factor is equivalent to the second-moment.
|
|
template_norm = sqrt(np.sum(template**2))
|
|
else:
|
|
template_norm = sqrt((template_mean ** 2)) / sqrt(inv_area)
|
|
|
|
# define window of template size in squared integral image
|
|
cdef int i, j
|
|
cdef float num, window_sum2, window_mean2, normed, t,
|
|
# move window through convolution results, normalizing in the process
|
|
for i in range(result.shape[0]):
|
|
for j in range(result.shape[1]):
|
|
num = result[i, j]
|
|
i_end = i + template.shape[0]
|
|
j_end = j + template.shape[1]
|
|
|
|
window_mean2 = 0
|
|
if method == 'norm-coeff':
|
|
t = sum_integral(integral_sum, i, j, i_end, j_end)
|
|
window_mean2 = t * t * inv_area
|
|
num -= t*template_mean
|
|
# calculate squared template window sum in the image
|
|
window_sum2 = sum_integral(integral_sqr, i, j, i_end, j_end)
|
|
|
|
normed = sqrt(window_sum2 - window_mean2) * template_norm
|
|
# enforce some limits
|
|
if fabs(num) < normed:
|
|
num /= normed
|
|
elif fabs(num) < normed*1.125:
|
|
if num > 0:
|
|
num = 1
|
|
else:
|
|
num = -1
|
|
else:
|
|
num = 0
|
|
result[i, j] = num
|
|
return result
|
|
|