【python学习笔记】25:scipy中值滤波

本文介绍了中值滤波技术的基本原理及应用,通过Python代码示例展示了如何在一维和二维数据上实现中值滤波,并对图像进行中值滤波处理以去除噪声。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

中值滤波技术能有效抑制噪声,通过把数字图像中一点的值用该点周围的各点值的中位数来代替,让这些值接近,以消除原图像中的噪声。

*模拟中值滤波

>>> import random
>>> import numpy as np
>>> import scipy.signal as signal
>>> x=np.arange(0,100,10)
>>> random.shuffle(x)
>>> x
array([70, 80, 30, 20, 10, 90,  0, 60, 40, 50])
>>> signal.medfilt(x,3) #一维中值滤波
array([ 70.,  70.,  30.,  20.,  20.,  10.,  60.,  40.,  50.,  40.])
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

signal的medfilt()方法传入两个参数,第一个参数是要作中值滤波的信号,第二个参数是邻域的大小(奇数)。如邻域为3即是每个点自己和左右各一个点成为一个邻域。在每个位置的邻域中选取中位数替换这个位置的数,也就是该函数的返回值数组。如果邻域中出现没有元素的位置,那么以0补齐。

>>> x=np.random.randint(1,1000,(4,4))
>>> x
array([[ 31,  33, 745, 483],
       [331, 469, 804, 479],
       [235, 487, 244, 982],
       [857, 114, 167, 174]])
>>> signal.medfilt(x,(3,3)) #二维中值滤波
array([[   0.,   33.,  469.,    0.],
       [  33.,  331.,  483.,  479.],
       [ 235.,  331.,  469.,  174.],
       [   0.,  167.,  167.,    0.]])
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

二维中值滤波还可以用signal.medfilt2d(),速度较快,但只支持int8,float32和float64。

*对图像中值滤波
测试图像如下,是在网上找的一个200*200像素的头像。
这里写图片描述

import numpy as np
from PIL import Image
import scipy.signal as signal

im=Image.open('test.jpg') #读入图片并建立Image对象im
data=[] #存储图像中所有像素值的list(二维)
width,height=im.size #将图片尺寸记录下来

#读取图像像素的值
for h in range(height): #对每个行号h
    row=[] #记录每一行像素
    for w in range(width): #对每行的每个像素列位置w
        value=im.getpixel((h,w)) #用getpixel读取这一点像素值
        row.append(value)#把它加到这一行的list中去
    data.append(row) #把记录好的每一行加到data的子list中去,就建立了模拟的二维list

data=signal.medfilt(data,kernel_size=3) #二维中值滤波
data=np.int32(data) #转换为int类型,以使用快速二维滤波

#创建并保存结果
for h in range(height): #对每一行
    for w in range(width): #对该行的每一个列号
        im.putpixel((h,w),tuple(data[h][w])) #将data中该位置的值存进图像,要求参数为tuple

im.save('result.jpg')#存储
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26

董老师课程中写的代码在我的电脑(可能是版本的问题)上运行会报错,我修改了之后的如上,这样可以进行中值滤波了,但是图像的颜色不知道为什么也发生了改变(不清楚是否是正常的)。得到下图。
这里写图片描述
如果把int32()改成int8()的话,图像又会被反相(似乎int8()不能把每一位转回表示0~255的三元组)。
如果想滤波得更”严重”一些,只要修改signal.medfilt的参数kernel_size更大一些就可以了,如把它改成5的时候图像如下。
这里写图片描述
这个时候竟然恢复了黑色,这样我认为之前出现绿色的情况也是合理的,可能是该图像本身就有许多绿色的成分在里面(在黑和白之间不易分辨)吧。

### 如何在 Python 中对一维信号进行中值滤波处理 对于在一维信号上执行中值滤波的任务,`scipy.signal.medfilt` 是一种常用方法。此函数可以应用于一维或多维数组,在指定窗口内计算中位数值来替代中心位置的原始值从而达到平滑效果的目的。 需要注意的是,当使用 `medfilt` 函数时,输入的一维数组不应被显式地转换成列向量形式(即形状为 (N, 1) 的二维数组),而应该保持其作为扁平化的一维结构(长度 N)。这是因为 SciPy 库中的某些版本可能无法正确处理具有特定维度布局的数据[^1]。 下面是一个简单的例子展示如何利用 `scipy.signal.medfilt` 对一维噪声信号实施中值滤波: ```python import numpy as np from scipy import signal import matplotlib.pyplot as plt # 创建带随机噪音的一维测试信号 np.random.seed(0) t = np.linspace(0, 1, 200, endpoint=False) sig = np.sin(2 * np.pi * t) noise = np.random.normal(.5, .5, sig.shape) noisy_sig = sig + noise # 定义并应用中值滤波器 filtered_signal = signal.medfilt(noisy_sig, kernel_size=9) # 绘制原图与过滤后的对比图像 plt.figure(figsize=(8, 6)) plt.plot(t, noisy_sig, label='Noisy Signal') plt.plot(t, filtered_signal, linewidth=3, label='Filtered Signal') plt.legend() plt.show() ``` 这段代码首先构建了一个带有高斯白噪声污染的理想正弦曲线样本;接着通过调用 `signal.medfilt()` 方法对该受扰动的时间序列进行了中值滤波操作,并指定了核大小为 9 来控制局部邻域范围内的统计特性估计过程;最后借助 Matplotlib 展示了未经处理以及经过滤波净化之后的结果比较图表。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值