基于python 实现对图片进行SVD

实现对图片进行SVD

之前在学习截断式SVD的时候, 不知道为什么截断式SVD后能够近似原矩阵? 或者说奇异值究竟有什么含义?
看了以下知乎的答案后, 有了很形象的理解
奇异值的物理意义是什么? - 郑宁的回答 - 知乎

这里参考基于Python查看SVD压缩图片的效果这篇博客后实现了截断式SVD

from PIL import Image
import numpy as np
import matplotlib.pyplot as plt 
import matplotlib.image as mpimg 

def get_approx_matrix(u, sigma, v, rank): #rank 表示截断式SVD的保留的奇异值个数
    m = len(u)
    n = len(v)
    a = np.zeros((m, n))
    k = 0
    while k < rank:
        uk = u[:, k].reshape(m, 1)
        vk = v[k].reshape(1, n)
        a += sigma[k] * np.dot(uk, vk)
        k += 1
    a[a < 0] = 0
    a[a > 255] = 255
    return a.astype("uint8")

def get_svd_image(file_path):
    name, suffix = file_path.split(".")
    img = Image.open(file_path, 'r')
    a = np.array(img)
    u0, sigma0, v0 = np.linalg.svd(a[:, :, 0])
    u1, sigma1, v1 = np.linalg.svd(a[:, :, 1])
    u2, sigma2, v2 = np.linalg.svd(a[:, :, 2])
    print("origin image \n red rank:",len(sigma0), " green_rank:", 
          len(sigma1)," blue_rank:", len(sigma2))
    for rank in np.arange(5, 50, 5):
        red_matrix = get_approx_matrix(u0, sigma0, v0, rank)
        green_matrix = get_approx_matrix(u1, sigma1, v1, rank)
        blue_matrix = get_approx_matrix(u2, sigma2, v2, rank)
        I = np.stack((red_matrix, green_matrix, blue_matrix), 2)
        Image.fromarray(I).save(name + str(rank) + "." +suffix)    


get_svd_image("gakki.png")
plt.figure(figsize=(20,20))
for i in range(1, 10, 1):
    ax = plt.subplot(3,3,i)
    img = mpimg.imread("gakki_" + str(i*5) + ".png") 
    ax.imshow(img)
    ax.set_title("rank:"+str(i*5))
    ax.set_axis_off()
plt.savefig('gakki_svd_compare.png')
plt.show()

下面就是gakki的效果图啦
这里写图片描述

猜你喜欢

转载自blog.csdn.net/xfzero/article/details/80385354