10

我最近一直在尝试找到一种快速有效的方法来使用 Python 语言在两个数组之间执行互相关检查。经过一番阅读,我发现了这两个选项:

  1. NumPy.correlate()方法在涉及大型数组时太慢了。
  2. cv.MatchTemplate()方法似乎要快得多。

出于显而易见的原因,我选择了第二个选项。我尝试执行以下代码:

import scipy
import cv

image = cv.fromarray(scipy.float32(scipy.asarray([1,2,2,1])),allowND=True)
template = cv.fromarray(scipy.float32(scipy.asarray([2,2])),allowND=True)
result = cv.fromarray(scipy.float32(scipy.asarray([0,0,0])),allowND=True)
cv.MatchTemplate(image,template,result,cv.CV_TM_CCORR)

尽管此代码假设非常简单,但它会引发下一个错误:

OpenCV Error: Bad flag (parameter or structure field) (Unrecognized or unsupported array type) in cvGetMat, file /builddir/build/BUILD/OpenCV-2.1.0/src/cxcore/cxarray.cpp, line 2476
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
cv.error: Unrecognized or unsupported array type

经过几个小时的令人沮丧的尝试,我仍然卡住了!有人有什么建议吗?

顺便说一句,这是我的 Python 版本输出:

Python 2.7 (r27:82500, Sep 16 2010, 18:03:06) 
[GCC 4.5.1 20100907 (Red Hat 4.5.1-3)] on linux2
Type "help", "copyright", "credits" or "license" for more information.

谢谢你们!

4

2 回答 2

22

您不太可能比使用基于 fft 的相关方法更快。

import numpy
from scipy import signal

data_length = 8192

a = numpy.random.randn(data_length)
b = numpy.zeros(data_length * 2)

b[data_length/2:data_length/2+data_length] = a # This works for data_length being even

# Do an array flipped convolution, which is a correlation.
c = signal.fftconvolve(b, a[::-1], mode='valid') 

# Use numpy.correlate for comparison
d = numpy.correlate(a, a, mode='same')

# c will be exactly the same as d, except for the last sample (which 
# completes the symmetry)
numpy.allclose(c[:-1], d) # Should be True

现在进行时间比较:

In [12]: timeit b[data_length/2:data_length/2+data_length] = a; c = signal.fftconvolve(b, a[::-1], mode='valid')
100 loops, best of 3: 4.67 ms per loop

In [13]: timeit d = numpy.correlate(a, a, mode='same')
10 loops, best of 3: 69.9 ms per loop

如果您可以处理循环相关,则可以删除副本。时间差会随着增加而data_length增加。

于 2012-09-07T19:58:10.670 回答
3

我相信您的代码失败了,因为 OpenCV 期望图像为 uint8 而不是 float32 格式。您可能会发现 cv2 python 界面使用起来更直观(ndarray 和 CV 图像格式之间的自动转换)。

至于关联的速度,您可以尝试使用快速 fft 实现(FFTW 有一个 python 包装器:pyfftw)。

于 2012-09-10T08:08:50.977 回答