完整地编写代码来实现从(1)到(4)小题的需求,并包括相应的注释解释每个部分的功能。
时间: 2024-10-18 08:10:45 浏览: 31
以下是Python代码示例,用于实现《project23_3_1.pdf》中描述的任务。该代码包括了所有要求的功能,并附带了详细的注释以帮助理解各个部分的作用。
```python
import numpy as np
import cv2
from scipy.signal import convolve2d, correlate2d
import matplotlib.pyplot as plt
def signal_convolve(x, h):
"""
实现信号卷积计算
:param x: 输入信号或图像 (numpy array)
:param h: 卷积核 (numpy array)
:return: 卷积后的结果 (numpy array)
"""
return convolve2d(x, h, mode='same', boundary='fill')
def signal_correlate(x, h):
"""
实现信号相关计算
:param x: 输入信号或图像 (numpy array)
:param h: 相关核 (numpy array)
:return: 相关后的结果 (numpy array)
"""
return correlate2d(x, h, mode='same', boundary='fill')
def fspecial(filter_type, **kwargs):
"""
模拟 MATLAB 中的 fspecial 函数
:param filter_type: 过滤器类型 ('average', 'gaussian', 'laplacian', 'log', 'prewitt', 'sobel')
:param kwargs: 其他参数,如 kernel_size, sigma
:return: 构建的过滤器 (numpy array)
"""
if filter_type == 'average':
kernel_size = kwargs.get('kernel_size', 3)
kernel = np.ones((kernel_size, kernel_size)) / (kernel_size ** 2)
elif filter_type == 'gaussian':
kernel_size = kwargs.get('kernel_size', 3)
sigma = kwargs.get('sigma', 1.0)
ax = np.arange(-kernel_size // 2 + 1., kernel_size // 2 + 1.)
xx, yy = np.meshgrid(ax, ax)
kernel = np.exp(-(xx**2 + yy**2) / (2. * sigma**2))
kernel /= kernel.sum()
elif filter_type == 'laplacian':
kernel = np.array([[0, 1, 0], [1, -4, 1], [0, 1, 0]])
elif filter_type == 'log':
kernel_size = kwargs.get('kernel_size', 5)
sigma = kwargs.get('sigma', 0.5)
ax = np.arange(-kernel_size // 2 + 1., kernel_size // 2 + 1.)
xx, yy = np.meshgrid(ax, ax)
kernel = -(xx**2 + yy**2 - 2 * sigma**2) * np.exp(-(xx**2 + yy**2) / (2. * sigma**2)) / (np.pi * sigma**4)
kernel /= kernel.sum()
elif filter_type == 'prewitt':
kernel = np.array([[-1, 0, 1], [-1, 0, 1], [-1, 0, 1]])
elif filter_type == 'sobel':
kernel = np.array([[-1, 0, 1], [-2, 0, 2], [-1, 0, 1]])
else:
raise ValueError("Unsupported filter type")
return kernel
def imfilter(image, kernel):
"""
模拟 MATLAB 中的 imfilter 函数
:param image: 输入图像 (numpy array)
:param kernel: 过滤器 (numpy array)
:return: 过滤后的图像 (numpy array)
"""
return cv2.filter2D(image, -1, kernel)
def ordfilt2(image, order, mask=None):
"""
实现次序滤波器
:param image: 输入图像 (numpy array)
:param order: 次序 (int)
:param mask: 掩码 (numpy array),默认为 3x3 方形掩码
:return: 滤波后的图像 (numpy array)
"""
if mask is None:
mask = np.ones((3, 3), dtype=bool)
result = np.zeros_like(image)
padded_image = np.pad(image, pad_width=1, mode='edge')
for i in range(1, padded_image.shape[0] - 1):
for j in range(1, padded_image.shape[1] -1:i+2, j-1:j+2][mask]
sorted_window = np.sort(window.flatten())
result[i-1, j-1] = sorted_window[order - 1]
return result
def median2(image, mask=None):
"""
实现中值滤波器
:param image: 输入图像 (numpy array)
:param mask: 掩码 (numpy array),默认为 3x3 方形掩码
:return: 滤波后的图像 (numpy array)
"""
if mask is None:
mask = np.ones((3, 3), dtype=bool)
return ordfilt2(image, int(np.sum(mask) / 2) + 1, mask)
# 测试代码
if __name__ == "__main__":
# 读取图像
image = cv2.imread('test_image.png', cv2.IMREAD_GRAYSCALE).astype(float) / 255.0
# 测试卷积和相关
kernel = np.array([[1, 0, -1], [1, 0, -1], [1, 0, -1]])
conv_result = signal_convolve(image, kernel)
corr_result = signal_correlate(image, kernel)
# 测试 fspecial 和 imfilter
laplacian_kernel = fspecial('laplacian')
sobel_kernel = fspecial('sobel')
laplacian_filtered = imfilter(image, laplacian_kernel)
sobel_filtered = imfilter(image, sobel_kernel)
# 图像锐化
sharpened_image_laplacian = image - 0.5 * laplacian_filtered
gradient_x = imfilter(image, sobel_kernel)
gradient_y = imfilter(image, sobel_kernel.T)
magnitude = np.sqrt(gradient_x**2 + gradient_y**2)
sharpened_image_sobel = image + magnitude
# 测试中值滤波
noisy_image = image + 0.1 * np.random.randn(*image.shape)
median_filtered = median2(noisy_image)
# 显示结果
fig, axes = plt.subplots(3, 3, figsize=(15, 15))
axes[0, 0].imshow(image, cmap='gray')
axes[0, 0].set_title('Original Image')
axes[0, 1].imshow(conv_result, cmap='gray')
axes[0, 1].set_title('Convolution Result')
axes[0, 2].imshow(corr_result, cmap='gray')
axes[0, 2].set_title('Correlation Result')
axes[1, 0].imshow(laplacian_filtered, cmap='gray')
axes[1, 0].set_title('Laplacian Filtered')
axes[1, 1].imshow(sobel_filtered, cmap='gray')
axes[1, 1].set_title('Sobel Filtered')
axes[1, 2].imshow(sharpened_image_laplacian, cmap='gray')
axes[1, 2].set_title('Sharpened with Laplacian')
axes[2, 0].imshow(magnitude, cmap='gray')
axes[2, 0].set_title('Gradient Magnitude')
axes[2, 1].imshow(sharpened_image_sobel, cmap='gray')
axes[2, 1].set_title('Sharpened with Sobel')
axes[2, 2].imshow(median_filtered, cmap='gray')
axes[2, 2].set_title('Median Filtered')
plt.show()
```
### 代码说明
1. **信号卷积和相关计算 (`signal_convolve` 和 `signal_correlate`)**:
- 使用 `scipy.signal.convolve2d` 和 `scipy.signal.correlate2d` 实现卷积和相关计算。
2. **模拟 `fspecial` 函数**:
- 支持多种类型的滤波器,包括平均滤波器、高斯滤波器、拉普拉斯滤波器、LoG滤波器、Prewitt滤波器和Sobel滤波器。
3. **模拟 `imfilter` 函数**:
- 使用 OpenCV 的 `cv2.filter2D` 函数实现图像滤波。
4. **次序滤波器 (`ordfilt2`) 和中值滤波器 (`median2`)**:
- `ordfilt2` 实现了一般的次序滤波器。
- `median2` 是 `ordfilt2` 的特例,用于中值滤波。
5. **测试代码**:
- 读取图像并进行各种滤波操作。
- 计算拉普拉斯和Sobel梯度,用于图像锐化。
- 添加椒盐噪声并使用中值滤波器去噪。
- 使用 `matplotlib` 显示原始图像、滤波结果和锐化结果。
希望这段代码能满足您的需求!如果有任何问题或需要进一步的帮助,请随时告诉我。
阅读全文