注意
转到末尾下载完整的示例代码。或通过 Binder 在浏览器中运行此示例
相位解包裹#
某些信号只能以模 2*pi 的方式观察到,这也可以应用于二维和三维图像。在这些情况下,需要相位解包裹来恢复底层未包裹的信号。在此示例中,我们将演示在 skimage
中实现的算法 [1] 如何处理此类问题。一维、二维和三维图像都可以使用 skimage 解包裹。这里我们将演示二维情况下的相位解包裹。
import numpy as np
from matplotlib import pyplot as plt
from skimage import data, img_as_float, color, exposure
from skimage.restoration import unwrap_phase
# Load an image as a floating-point grayscale
image = color.rgb2gray(img_as_float(data.chelsea()))
# Scale the image to [0, 4*pi]
image = exposure.rescale_intensity(image, out_range=(0, 4 * np.pi))
# Create a phase-wrapped image in the interval [-pi, pi)
image_wrapped = np.angle(np.exp(1j * image))
# Perform phase unwrapping
image_unwrapped = unwrap_phase(image_wrapped)
fig, ax = plt.subplots(2, 2, sharex=True, sharey=True)
ax1, ax2, ax3, ax4 = ax.ravel()
fig.colorbar(ax1.imshow(image, cmap='gray', vmin=0, vmax=4 * np.pi), ax=ax1)
ax1.set_title('Original')
fig.colorbar(ax2.imshow(image_wrapped, cmap='gray', vmin=-np.pi, vmax=np.pi), ax=ax2)
ax2.set_title('Wrapped phase')
fig.colorbar(ax3.imshow(image_unwrapped, cmap='gray'), ax=ax3)
ax3.set_title('After phase unwrapping')
fig.colorbar(ax4.imshow(image_unwrapped - image, cmap='gray'), ax=ax4)
ax4.set_title('Unwrapped minus original')
Text(0.5, 1.0, 'Unwrapped minus original')
解包裹过程接受掩码数组,并且还可以选择性地假设循环边界来连接图像的边缘。在下面的示例中,我们研究一个简单的相位斜坡,该斜坡通过掩盖图像的一行而被分成两部分。
# Create a simple ramp
image = np.ones((100, 100)) * np.linspace(0, 8 * np.pi, 100).reshape((-1, 1))
# Mask the image to split it in two horizontally
mask = np.zeros_like(image, dtype=bool)
mask[image.shape[0] // 2, :] = True
image_wrapped = np.ma.array(np.angle(np.exp(1j * image)), mask=mask)
# Unwrap image without wrap around
image_unwrapped_no_wrap_around = unwrap_phase(image_wrapped, wrap_around=(False, False))
# Unwrap with wrap around enabled for the 0th dimension
image_unwrapped_wrap_around = unwrap_phase(image_wrapped, wrap_around=(True, False))
fig, ax = plt.subplots(2, 2)
ax1, ax2, ax3, ax4 = ax.ravel()
fig.colorbar(ax1.imshow(np.ma.array(image, mask=mask), cmap='rainbow'), ax=ax1)
ax1.set_title('Original')
fig.colorbar(ax2.imshow(image_wrapped, cmap='rainbow', vmin=-np.pi, vmax=np.pi), ax=ax2)
ax2.set_title('Wrapped phase')
fig.colorbar(ax3.imshow(image_unwrapped_no_wrap_around, cmap='rainbow'), ax=ax3)
ax3.set_title('Unwrapped without wrap_around')
fig.colorbar(ax4.imshow(image_unwrapped_wrap_around, cmap='rainbow'), ax=ax4)
ax4.set_title('Unwrapped with wrap_around')
plt.tight_layout()
plt.show()
在上面的图中,掩码行可以看作是穿过图像的白色线条。底部行中两个解包裹图像之间的差异很明显:没有解包裹(左下),掩码边界之上和之下的区域根本不相互作用,导致两个区域之间的偏移为任意整数倍的 2 pi。我们也可以将这些区域解包裹为两个单独的图像。当为垂直方向启用环绕时(右下),情况会发生变化:解包裹路径现在允许从图像的底部传递到顶部,反之亦然,从而有效地提供了一种确定两个区域之间偏移的方法。
参考资料#
脚本的总运行时间:(0 分钟 1.315 秒)