数字图像学笔记 —— 17. 图像退化与复原(自适应滤波之「最小二乘方滤波」)

news/2025/3/14 16:39:45/

文章目录

  • 维纳滤波的缺点
  • 约束最小二乘方滤波
  • 给一个实际例子吧

维纳滤波的缺点

维纳滤波(Wiener Filter),虽然是一种非常强大的退化图像还原算法,但是从实验过程我们也发现它存在着致命的缺陷,那就是要求输入退化系统的 F(u,v)F(u, v)F(u,v) 是已知的。分析维纳滤波的理论,我们发现它是通过给未知的退化系统 H(u,v)H(u, v)H(u,v) 输入原始数据 F(u,v)F(u, v)F(u,v) 后得到退化后的数据 G(u,v)G(u, v)G(u,v),然后通过梯度下降算法与约束条件 MSE=(F^−F)2MSE = (\hat F - F)^2MSE=(F^F)2 找出最接近 H(u,v)H(u, v)H(u,v) 的近似解 H^(u,v)\hat H(u, v)H^(u,v)

所以,在求解 H^(u,v)\hat H(u, v)H^(u,v) 的过程中,它要求我们必须要预先知道输入退化系统 H(u,v)H(u, v)H(u,v) 前的图像 F(u,v)F(u, v)F(u,v) 的信息。但是我们在实际生活中,不一定都能拥有这样的条件,可以让我们顺利的找出 H^(u,v)\hat H(u, v)H^(u,v),比方说对老照片的恢复。

在这里插入图片描述

拍摄时间未详的居里夫人照,图片来自网络。

对于上述图片来说,存在很明显的退化,而且我们也没有办法回到过去拍一张高清的照片,所以要想让照片得到修复,获得较好的视觉感受,维纳滤波在这里就无法使用了。

所以后续的研究者提出了一种仅依赖均值与方差便可以估算出最好复原效果的 「约束最小二乘方滤波(Constrained Least Squares Filtering)」。接下来我们来看看它是如何展现「化腐朽为神奇」的力量。

约束最小二乘方滤波

先从退化模型的一般形式(频率空间)出发,我们有如下公式:

G(u,v)=H(u,v)F(u,v)+N(u,v)G(u, v) = H(u, v) F(u, v) + N(u, v) G(u,v)=H(u,v)F(u,v)+N(u,v)

这是因为噪声函数 D(u,v)D(u, v)D(u,v) 通常被认为与输入图像 F(u,v)F(u, v)F(u,v) 相互独立,即 N(u,v)N(u, v)N(u,v) 是与 F(u,v)F(u, v)F(u,v) 不相关的噪声函数。因此,这个模型可以被看作是信号 F(u,v)F(u, v)F(u,v) 通过退化系统 H(u,v)H(u, v)H(u,v) 得到观测结果 G(u,v)G(u, v)G(u,v) 后再加上噪声 N(u,v)N(u, v)N(u,v)

接下来,我们可以使用最小二乘法来估计未知的退化函数 H(u,v)H(u, v)H(u,v)。最小二乘法是一种优化方法,可以通过最小化误差平方和来得到最优解。具体来说,对于给定的观测结果 G(u,v)G(u, v)G(u,v) 和输入图像 F(u,v)F(u, v)F(u,v),我们可以计算出它们之间的误差平方和:

E=∑u=0M−1∑v=0N−1∣G(u,v)2−H(u,v)F(u,v)∣2E = \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | G(u, v)^2 - H(u,v)F(u,v)|^2 E=u=0M1v=0N1G(u,v)2H(u,v)F(u,v)2

我们的目标是找到一个 H(u,v)H(u,v)H(u,v),使得误差平方和 EEE 最小化。因此,我们可以通过求解下面的优化问题来得到最优的 H(u,v)H(u,v)H(u,v)

min⁡H(u,v)E=∑u=0M−1∑v=0N−1∣G(u,v)2−H(u,v)F(u,v)∣2\min_{H(u,v)}E = \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | G(u, v)^2 - H(u,v)F(u,v)|^2 H(u,v)minE=u=0M1v=0N1G(u,v)2H(u,v)F(u,v)2

然而,我们需要加上一些约束条件,以防止退化函数 H(u,v)H(u,v)H(u,v) 变得过于复杂,从而导致过度拟合(overfitting)。一个常见的约束条件是 H(u,v)H(u,v)H(u,v) 的均值为 111,方差为 σH2\sigma^2_HσH2,即:

1MN∑u=0M−1∑v=0N−1H(u,v)=1\frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} H(u, v) = 1 MN1u=0M1v=0N1H(u,v)=1

1MN∑u=0M−1∑v=0N−1∣H(u,v)−1MN∑i=0M−1∑j=0N−1H(i,j)∣2=σH2\frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | H(u,v) - \frac{1}{MN} \sum_{i=0}^{M-1}\sum_{j=0}^{N-1}H(i, j) |^2 = \sigma^2_{H} MN1u=0M1v=0N1H(u,v)MN1i=0M1j=0N1H(i,j)2=σH2

这个约束条件的意义是,我们期望退化函数 H(u,v)H(u,v)H(u,v) 的均值为 111,这是因为退化函数的总能量应该保持不变。同时,方差 σH2\sigma^2_HσH2 用于限制 H(u,v)H(u,v)H(u,v) 的复杂度,从而避免过度拟合。

根据上述约束条件,我们可以得到一个带有约束的最小二乘优化问题:

min⁡H(u,v)E=∑u=0M−1∑v=0N−1∣G(u,v)−H(u,v)F(u,v)∣2\min_{H(u,v)} E = \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | G(u,v) - H(u, v)F(u, v) |^2 H(u,v)minE=u=0M1v=0N1G(u,v)H(u,v)F(u,v)2

s.t.1MN∑u=0M−1∑v=0N−1H(u,v)=1s.t. \frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} H(u, v) = 1 s.t.MN1u=0M1v=0N1H(u,v)=1

1MN∑u=0M−1∑v=0N−1∣H(u,v)−1MN∑i=0M−1∑j=0N−1H(i,j)∣2=σH2\frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | H(u, v) - \frac{1}{MN} \sum_{i=0}^{M-1} \sum_{j=0}^{N-1} H(i, j) |^2 = \sigma_{H}^{2} MN1u=0M1v=0N1H(u,v)MN1i=0M1j=0N1H(i,j)2=σH2

上面这个优化问题可以通过拉格朗日乘子法来求解。我们首先构造一个拉格朗日函数:

L(H,λ1,λ2)=∑u=0M−1∑v=0N−1∣G(u,v)−H(u,v)F(u,v)∣2+λ1(1MN∑u=0M−1∑v=0N−1H(u,v)−1)+λ2(1MN∑u0M−1∑v=0N−1∣H(u,v)−1MN∑i=0M−1∑j=0N−1H(i,j)∣2−σH2)L(H, \lambda_1, \lambda_2) = \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | G(u, v) - H(u, v)F(u, v)|^2 + \lambda_1 (\frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} H(u, v) - 1) + \lambda_2 (\frac{1}{MN} \sum_{u_0}^{M-1} \sum_{v=0}^{N-1} | H(u, v) - \frac{1}{MN} \sum_{i=0}^{M-1} \sum_{j=0}^{N-1} H(i, j)|^2 - \sigma_H^2) L(H,λ1,λ2)=u=0M1v=0N1G(u,v)H(u,v)F(u,v)2+λ1(MN1u=0M1v=0N1H(u,v)1)+λ2(MN1u0M1v=0N1H(u,v)MN1i=0M1j=0N1H(i,j)2σH2)

其中 λ1\lambda_1λ1λ2\lambda_2λ2 是拉格朗日乘子。接下来,我们对 L(H,λ1,λ2)L(H, \lambda_1, \lambda_2)L(H,λ1,λ2) 分别对 H(u,v)H(u,v)H(u,v)λ1\lambda_1λ1λ2\lambda_2λ2 求偏导数,并令它们等于 000,可以得到下面的一组方程:

∂L∂λ1=1MN∑u=0M−1∑v=0N−1H(u,v)−1=0\frac{\partial L}{\partial \lambda_1} = \frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} H(u, v) - 1 =0 λ1L=MN1u=0M1v=0N1H(u,v)1=0

∂L∂H(u,v)=−2F(u,v)(G(u,v)−H(u,v)F(u,v))+λ1+2λ2(H(u,v)−1MN∑i=0M−1∑j=0N−1H(i,j))=0\frac{\partial L}{\partial H(u, v)} = -2 F(u, v)( G(u, v) - H(u, v)F(u, v) ) + \lambda_1 + 2 \lambda_2 ( H(u, v) - \frac{1}{MN} \sum_{i=0}^{M-1} \sum_{j=0}^{N-1} H(i, j)) = 0 H(u,v)L=2F(u,v)(G(u,v)H(u,v)F(u,v))+λ1+2λ2(H(u,v)MN1i=0M1j=0N1H(i,j))=0

∂L∂λ2=1MN∑u=0M−1∑v=0N−1∣H(u,v)−1MN∑i=0M−1∑j=0N−1H(i,j)∣2−σH2=0\frac{\partial L}{ \partial \lambda_2} = \frac{1}{MN} \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | H(u, v) - \frac{1}{MN} \sum_{i=0}^{M-1} \sum_{j=0}^{N-1} H(i ,j) |^2 - \sigma_H^2 = 0 λ2L=MN1u=0M1v=0N1H(u,v)MN1i=0M1j=0N1H(i,j)2σH2=0

解这个方程组可以得到最优的 H(u,v)H(u,v)H(u,v)。具体地,我们可以先把 λ1\lambda_1λ1λ2\lambda_2λ2 消去,然后将 H(u,v)H(u,v)H(u,v) 写成矩阵形式 H=[h1,h2,...,hL]H = [h_1, h_2, ..., h_L]H=[h1,h2,...,hL],其中 L=MNL=MNL=MN。接着,将偏导数为 000 的方程写成矩阵形式 Ah=bA\textbf{h} = \textbf{b}Ah=b,其中 AAAb\textbf{b}b 可以根据上述方程计算得到,h\textbf{h}h 是一个列向量,它包含了所有的 hih_ihi。最后,我们可以通过求解下面的矩阵方程来得到最优的 h\textbf{h}h

(ATA+λI)h=ATg(A^{T} A + \lambda I) \mathbf h = A^T \mathbf g (ATA+λI)h=ATg

其中 g\textbf{g}gG(u,v)G(u,v)G(u,v) 的列向量形式,III 是单位矩阵,λ\lambdaλ 是一个正则化参数,用于平衡拟合误差和模型复杂度。

最终,我们可以得到最优的 H(u,v)H(u,v)H(u,v)

H(u,v)=huN+vH(u, v) = h_{uN + v} H(u,v)=huN+v

其中 hih_ihi 是列向量 h\textbf{h}h 中的第 iii 个元素。根据最优的 H(u,v)H(u,v)H(u,v),我们可以得到修复后的图像 Frestored(u,v)F_{restored}(u,v)Frestored(u,v)

Frestored(u,v)=F−1{G(u,v)/H(u,v)}F_{restored} (u, v) = \mathcal{F}^{-1} \{ G(u,v) / H(u,v) \} Frestored(u,v)=F1{G(u,v)/H(u,v)}

其中 F−1\mathcal{F}^{-1}F1 是傅里叶逆变换。由于在实际应用中,H(u,v)H(u,v)H(u,v) 可能会变得过于复杂,从而导致过度拟合。为了避免这种情况,我们可以使用正则化技术,例如 Tikhonov 正则化,来控制模型的复杂度。具体来说,我们可以将优化问题改写为:

min⁡H(u,v)E=∑u=0M−1∑v=0N−1∣G(u,v)−H(u,v)F(u,v)∣2+λ∑u=0M−1∑v=0N−1∣H(u,v)∣2\min_{H(u, v)} E = \sum_{u=0}^{M-1} \sum_{v=0}^{N-1} | G(u, v) - H(u, v)F(u, v) |^2 + \lambda \sum_{u = 0}^{M-1} \sum_{v=0}^{N-1} | H(u, v)|^2 H(u,v)minE=u=0M1v=0N1G(u,v)H(u,v)F(u,v)2+λu=0M1v=0N1H(u,v)2

其中 λ\lambdaλ 是一个正则化参数,用于控制 H(u,v)H(u,v)H(u,v) 的平滑度。这样,我们可以得到下面的矩阵方程:

(ATA+λI)h=ATg(A^T A + \lambda I) \textbf h = A^T \textbf g (ATA+λI)h=ATg

解这个方程可以得到带有正则化项的最优 H(u,v)H(u,v)H(u,v),从而得到修复后的图像 Frestored(u,v)F_{restored}(u,v)Frestored(u,v)

总之,约束最小二乘方滤波是一种非常强大的图像修复技术,它可以通过仅依赖于均值和方差的约束条件来估计未知的退化函数,并从退化的图像中恢复出尽可能接近原始图像的图像。相比于维纳滤波,它更具有实用性和适用性。

给一个实际例子吧

上述内容实在太多,但是OpenCV已经帮我们减少了很多工作量,所以就可以简化为几个关键函数,所以我们就可以这样去调用它们

def ConstrainedLeastSquaresFiltering(image_path):# Load the image as grayscaleimg = cv2.imread(image_path, cv2.IMREAD_GRAYSCALE)# Add Gaussian noise to the imagenoisy_img = img + np.random.normal(0, 20, size=img.shape)# Define the degradation matrixdegradation_matrix = np.array([[0.9, 0.1, 0], [0.1, 0.8, 0.1], [0, 0.1, 0.9]])# Define the constraint matrixconstraint_matrix = np.array([[0, 1, 0], [1, -4, 1], [0, 1, 0]])# Define the regularization parameteralpha = 0.1# Compute the inverse of the degradation matrixdegradation_matrix_inv = np.linalg.inv(degradation_matrix)# Compute the constrained least squares estimatecls_estimate = cv2.filter2D(noisy_img, -1, degradation_matrix_inv)cls_estimate = cls_estimate - alpha * cv2.filter2D(cls_estimate, -1, constraint_matrix)# Convert the image to uint8 and clip it to [0, 255]cls_estimate = np.clip(cls_estimate, 0, 255).astype(np.uint8)# Show the original image, the noisy image, and the restored imagecv2.imshow("Original Image", img)cv2.imshow("Noisy Image", noisy_img)cv2.imshow("Restored Image", cls_estimate)cv2.waitKey(0)cv2.destroyAllWindows()

那么自然输出的效果就变成下面这个样子了

在这里插入图片描述

怎么样,是不是很厉害的技术呢?


http://www.ppmy.cn/news/30262.html

相关文章

KNN学习报告

原理 KNN算法就是在其表征空间中,求K个最邻近的点。根据已知的这几个点对其进行分类。如果其特征参数只有一个,那么就是一维空间。如果其特征参数只有两个,那么就是二维空间。如果其特征参数只有三个,那么就是三维空间。如果其特征…

进程控制~

进程控制 (创建、终止,等待,程序替换) 进程创建: pid_t fork();父子进程,数据独有,代码共享,各有各的地址 pit_t vfork();父进程阻塞,直到子进程exit退出或者程序替换之…

shell基础(5)算数计算:运算语法、自增自减

文章目录1. shell算数运算的特点2. 运算符一览3. 运算语法3.1 整形运算3.2. 小数运算 ing4. 自增自减4.1. a与a4.2. 自加1. shell算数运算的特点 Shell 和其它编程语言不同,Shell 不能直接进行算数运算,必须使用数学计算命令。Shell只支持整数运算&#…

PDF 解析格式化输出 API 数据接口

PDF 解析格式化输出 API 数据接口 支持输出 TEXT HTML XML TAG,多种格式输出,超精准识别率。 1. 产品功能 通用的识别接口, 支持标准 PDF 文件解析;多种格式输出,支持 TEXT HTML XML TAG;HTML 包含完美排…

Pytorch处理数据与训练网络问题汇总(协同训练)

基础语法 模型训练 【Swin-Unet】官方代码预训练权重加载函数load_from() 实际上由于SwinUnet是一个encoder-decoder对称的结构,因此加载权重时,作者并没有像通常那样仅仅加载encoder部分而不加载decoder部分,而是同时将encoder的权重对称地…

滚动升级回滚

滚动升级回滚 ReplicationController 资源文件 apiVersion: v1 kind: ReplicationController metadata:name: kubia-v1labels:app: kubia spec:replicas: 3template:metadata:name: kubialabels:app: kubiaspec:containers:- image: luksa/kubia:v1name: nodejes --- apiVer…

【ONE·C || 文件操作】

总言 C语言:文件操作。    文章目录总言1、文件是什么?为什么需要文件?1.1、为什么需要文件?1.2、文件是什么?2、文件的打开与关闭2.1、文件指针2.2、文件打开和关闭:fopen、fclose2.3、文件使用方式3、文…

【Java】反射机制和代理机制

目录一、反射1. 反射概念2. 反射的应用场景3. 反射机制的优缺点4. 反射实战获取 Class 对象的四种方式二、代理机制1. 代理模式2. 静态代理3. 动态代理3.1 JDK动态代理机制1. 介绍2.JDK 动态代理类使用步骤3. 代码示例3.2 CGLIB 动态代理机制1.介绍2.CGLIB 动态代理类使用步骤3…