基于SVD图片压缩

基于SVD图片压缩原理:

图片其实就是数字矩阵,通过SVD将该矩阵降维,只使用其中的重要特征来表示该图片从而达到了压缩的目的。

数据集降维:

dim = data.T * U[:,:count] * dig.I # 降维

这里的dig为对角矩阵(需要利用原来svd返回的sigma向量构建矩阵,构建需要使用count这个值)。

U为svd返回的左奇异矩阵,count为我们指定的多少个奇异值,这也是dig矩阵的维数。

重构数据集:

redata = U[:, :count] * dig * VT[:count, :]  # 重构

这里的dig同样为对角矩阵(需要利用原来svd返回的dig向量构建矩阵,构建需要使用count这个值),VT为svd返回的右奇异矩阵,count为我们指定的多少个奇异值(可以按能量90%规则选取或者指定前多少个奇异值)

# *===================================*
# -*- coding: utf-8 -*-
# * Time : 2019-12-10 20:15
# * Author : zhangsf
# *===================================*
from numpy import mat, eye
import matplotlib.pyplot as plt
import matplotlib.image as mpimg
import numpy as np

path = './img/1.jpg'

img_eg = mpimg.imread(path)
print(img_eg.shape)
# 奇异值分解
img_temp = img_eg.reshape(600, 599 * 3)
U, sigma, VT = np.linalg.svd(img_temp)
# 在重构之前,依据前面的方法需要选择达到某个能量度的奇异值
cnt = sum(sigma)
print(cnt)
cnt90 = 0.9 * cnt  # 达到90%时的奇异总值
print("达到90%时的奇异总值",cnt90)
count = 100  # 选择前100个奇异值
cntN = sum(sigma[:count])

print("取前10个奇异值",sum(sigma[:10]))
print("取前50个奇异值",sum(sigma[:50]))
print("取前100个奇异值",sum(sigma[:100]))

# 重构矩阵
dig = mat(eye(count) * sigma[:count])  # 获得对角矩阵
# dim = data.T * U[:,:count] * dig.I # 降维 格外变量这里没有用
redata = U[:, :count] * dig * VT[:count, :]  # 重构
plt.imshow(redata, cmap='gray')  # 取灰
plt.title("nums of sigma = 100")
plt.show()  # 可以使用save函数来保存图片

原图效果

前5个奇异值的图片

前50个奇异值的图片

前100个奇异值

取前 50 个最大奇异值来重构图像时,已经非常清晰了。我们得到和原图差别不大的图像。也就是说,随着选择的奇异值的增加,重构的图像越来越接近原图像。

基于这个原理,奇异值分解可以用来进行图片压缩。

显然,所需存储量大大减小了。在需要存储许多高清图片,而存储空间有限的情况下,就可以利用 SVD,保留奇异值最大的若干项,舍去奇异值较小的项即可。

奇异值从大到小衰减得特别快,在很多情况下,前 10% 甚至 1% 的奇异值的和就占了全部的奇异值之和的 99% 以上了。这样在压缩图片的时候就可以只需要取部分就可以展示出原图片的效果。

 

### 基于SVD的像素压缩算法实现与原理 #### 1. SVD分解的基本原理 奇异值分解(Singular Value Decomposition, SVD)是一种矩阵分解方法,能够将任意一个矩阵 \( A \) 分解为三个矩阵的乘积形式:\( A = U\Sigma V^T \)[^3]。其中: - \( U \) 是一个正交矩阵,其列向量是矩阵 \( A \) 的左奇异向量。 - \( \Sigma \) 是一个对角矩阵,包含矩阵 \( A \) 的奇异值,按从大到小排列。 - \( V^T \) 是一个正交矩阵,其行向量是矩阵 \( A \) 的右奇异向量。 通过保留前 \( k \) 个最大的奇异值及其对应的奇异向量,可以实现对矩阵的有效近似,从而达到降维或压缩的目的。 #### 2. 图像压缩中的应用 在图像处理中,图像可以被表示为一个二维矩阵,每个元素代表图像的一个像素值。通过 SVD 分解,图像矩阵 \( A \) 可以被分解为 \( U\Sigma V^T \)[^3]。为了实现压缩,可以仅保留 \( k \) 个最大的奇异值及其对应的奇异向量,构造一个新的近似矩阵 \( A_k \),其计算公式为: \[ A_k = U_k \Sigma_k V_k^T \] 其中 \( U_k \)、\( \Sigma_k \) 和 \( V_k \) 分别是保留了前 \( k \) 个奇异值及对应奇异向量的子矩阵[^3]。 #### 3. Python 实现代码示例 以下是一个基于 Python 的 SVD 图像压缩实现: ```python import numpy as np from PIL import Image import matplotlib.pyplot as plt # 导入图像并转换为灰度图像 def load_image(image_path): img = Image.open(image_path).convert('L') # 转换为灰度图 return np.array(img) # 定义压缩函数 def pic_compress(k, pic_array): u, sigma, vt = np.linalg.svd(pic_array, full_matrices=False) compressed_sigma = np.diag(sigma[:k]) compressed_u = u[:, :k] compressed_vt = vt[:k, :] approx_matrix = compressed_u @ compressed_sigma @ compressed_vt return approx_matrix # 显示原始图像和压缩后的图像 def display_images(original, compressed): fig, axes = plt.subplots(1, 2, figsize=(10, 5)) axes[0].imshow(original, cmap='gray') axes[0].set_title("Original Image") axes[1].imshow(compressed, cmap='gray') axes[1].set_title(f"Compressed Image (k={k})") plt.show() # 主程序 if __name__ == "__main__": image_path = "example.jpg" # 替换为你的图片路径 original_image = load_image(image_path) k = 50 # 设置保留的奇异值数量 compressed_image = pic_compress(k, original_image) display_images(original_image, compressed_image) ``` #### 4. 压缩比与图像质量的关系 压缩比定义为原始数据量与压缩后数据量的比值。对于 SVD 压缩压缩比可以通过以下公式计算: \[ \text{压缩比} = \frac{\text{原始矩阵大小}}{\text{压缩后矩阵大小}} \] 压缩后矩阵大小为 \( k \times (m + n + 1) \),其中 \( m \) 和 \( n \) 分别是原始矩阵的行数和列数[^1]。随着 \( k \) 的减小,压缩比增大,但图像质量会下降。 #### 5. 总结 SVD 算法通过保留矩阵的主要特征信息实现图像压缩。尽管该方法可能损失部分细节,但在适当选择 \( k \) 的情况下,仍能保持较高的视觉质量[^3]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhangvalue

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值