2019独角兽企业重金招聘Python工程师标准>>>

简介

变分自编码器(Variational Autoencoder,VAE)是生成式模型(Generative Model)的一种,另一种常见的生成式模型是生成式对抗网络(Generative Adversarial Network,GAN)

这里我们介绍下VAE的原理,并用Keras实现

原理

我们经常会有这样的需求:根据很多个样本,学会生成新的样本

以MNIST为例,在看过几千张手写数字图片之后,我们能进行模仿,并生成一些类似的图片,这些图片在原始数据中并不存在,有一些变化但是看起来相似

换言之,需要学会数据x的分布,这样,根据数据的分布就能轻松地产生新样本

$$ P(X) $$

但数据分布的估计不是件容易的事情,尤其是当数据量不足的时候

可以使用一个隐变量z,由z经过一个复杂的映射得到x,并且假设z服从高斯分布

$$ x=f(z;\theta) $$

因此只需要学习隐变量所服从高斯分布的参数,以及映射函数,即可得到原始数据的分布

为了学习隐变量所服从高斯分布的参数,需要得到z足够多的样本

然而z的样本并不能直接获得,因此还需要一个映射函数(条件概率分布),从已有的x样本中得到对应的z样本

$$ z=Q(z|x) $$

这看起来和自编码器很相似,从数据本身,经编码得到隐层表示,经解码还原

但VAE和AE的区别如下:

  • AE中隐层表示的分布未知,而VAE中隐变量服从高斯分布
  • AE中学习的是encoder和decoder,VAE中还学习了隐变量的分布,包括高斯分布的均值和方差
  • AE只能从一个x,得到对应的重构x
  • VAE可以产生新的z,从而得到新的x,即生成新的样本

损失函数

除了重构误差,由于在VAE中我们假设隐变量z服从高斯分布,因此encoder对应的条件概率分布,应当和高斯分布尽可能相似

可以用相对熵,又称作KL散度(Kullback–Leibler Divergence),来衡量两个分布的差异,或者说距离,但相对熵是非对称

$$ D(f\parallel g)=\int f(x)\log\frac{f(x)}{g(x)}dx $$

实现

这里以MNIST为例,学习隐变量z所服从高斯分布的均值和方差两个参数,从而可以从新的z生成原始数据中没有的x

encoder和decoder各用两层全连接层,简单一些,主要为了说明VAE的实现

加载库

# -*- coding: utf-8 -*-import numpy as np
import matplotlib.pyplot as pltfrom keras.layers import Input, Dense, Lambda
from keras.models import Model
from keras import backend as K
from keras import objectives
from keras.datasets import mnist

定义一些常数

batch_size = 100
original_dim = 784
intermediate_dim = 256
latent_dim = 2
epochs = 50

encoder部分,两层全连接层,隐层表示包括均值和方差

x = Input(shape=(original_dim,))
h = Dense(intermediate_dim, activation='relu')(x)
z_mean = Dense(latent_dim)(h)
z_log_var = Dense(latent_dim)(h)

Lambda层不参与训练,只参与计算,用于后面产生新的z

def sampling(args):z_mean, z_log_var = argsepsilon = K.random_normal(shape=(batch_size, latent_dim), mean=0.)return z_mean + K.exp(z_log_var / 2) * epsilonz = Lambda(sampling, output_shape=(latent_dim,))([z_mean, z_log_var])

decoder部分,两层全连接层,x_decoded_mean为重构的输出

decoder_h = Dense(intermediate_dim, activation='relu')
decoder_mean = Dense(original_dim, activation='sigmoid')
h_decoded = decoder_h(z)
x_decoded_mean = decoder_mean(h_decoded)

自定义总的损失函数并编译模型

def vae_loss(x, x_decoded_mean):xent_loss = original_dim * objectives.binary_crossentropy(x, x_decoded_mean)kl_loss = -0.5 * K.sum(1 + z_log_var - K.square(z_mean) - K.exp(z_log_var), axis=-1)return xent_loss + kl_lossvae = Model(x, x_decoded_mean)
vae.compile(optimizer='rmsprop', loss=vae_loss)

加载数据并训练,CPU训练的速度还算能忍

(x_train, y_train), (x_test, y_test) = mnist.load_data()x_train = x_train.astype('float32') / 255.
x_test = x_test.astype('float32') / 255.
x_train = x_train.reshape((len(x_train), np.prod(x_train.shape[1:])))
x_test = x_test.reshape((len(x_test), np.prod(x_test.shape[1:])))vae.fit(x_train, x_train,shuffle=True,epochs=epochs,batch_size=batch_size,validation_data=(x_test, x_test))

定义一个encoder,看看MNIST中的数据在隐层中变成了什么样子

encoder = Model(x, z_mean)x_test_encoded = encoder.predict(x_test, batch_size=batch_size)
plt.figure(figsize=(6, 6))
plt.scatter(x_test_encoded[:, 0], x_test_encoded[:, 1], c=y_test)
plt.colorbar()
plt.show()

结果如下,说明在二维的隐层中,不同的数字被很好地分开了

再定义一个生成器,从隐层到输出,用于产生新的样本

decoder_input = Input(shape=(latent_dim,))
_h_decoded = decoder_h(decoder_input)
_x_decoded_mean = decoder_mean(_h_decoded)
generator = Model(decoder_input, _x_decoded_mean)

用网格化的方法产生一些二维数据,作为新的z输入到生成器,并将生成的x展示出来

n = 20
digit_size = 28
figure = np.zeros((digit_size * n, digit_size * n))
grid_x = np.linspace(-4, 4, n)
grid_y = np.linspace(-4, 4, n)for i, xi in enumerate(grid_x):for j, yi in enumerate(grid_y):z_sample = np.array([[yi, xi]])x_decoded = generator.predict(z_sample)digit = x_decoded[0].reshape(digit_size, digit_size)figure[(n - i - 1) * digit_size: (n - i) * digit_size,j * digit_size: (j + 1) * digit_size] = digitplt.figure(figsize=(10, 10))
plt.imshow(figure)
plt.show()

结果如下,和之前看到的隐层图是一致的,甚至能看到一些数字之间的过渡态

由于包含一些随机因素,所以每次生成的结果会存在一些差异

如果将全连接层换成CNN,应该可以得到更好的表示结果

拓展

掌握以上内容后,用相同的方法,可以在FashionMNIST这个数据集上再跑一遍,数据集规模和MNIST完全相同

只需改动四行即可

from keras.datasets import fashion_mnist(x_train, y_train), (x_test, y_test) = fashion_mnist.load_data()grid_x = np.linspace(-3, 3, n)
grid_y = np.linspace(-3, 3, n)

完整代码如下

# -*- coding: utf-8 -*-import numpy as np
import matplotlib.pyplot as pltfrom keras.layers import Input, Dense, Lambda
from keras.models import Model
from keras import backend as K
from keras import objectives
from keras.datasets import fashion_mnistbatch_size = 100
original_dim = 784
intermediate_dim = 256
latent_dim = 2
epochs = 50x = Input(shape=(original_dim,))
h = Dense(intermediate_dim, activation='relu')(x)
z_mean = Dense(latent_dim)(h)
z_log_var = Dense(latent_dim)(h)def sampling(args):z_mean, z_log_var = argsepsilon = K.random_normal(shape=(batch_size, latent_dim), mean=0.)return z_mean + K.exp(z_log_var / 2) * epsilonz = Lambda(sampling, output_shape=(latent_dim,))([z_mean, z_log_var])decoder_h = Dense(intermediate_dim, activation='relu')
decoder_mean = Dense(original_dim, activation='sigmoid')
h_decoded = decoder_h(z)
x_decoded_mean = decoder_mean(h_decoded)def vae_loss(x, x_decoded_mean):xent_loss = original_dim * objectives.binary_crossentropy(x, x_decoded_mean)kl_loss = -0.5 * K.sum(1 + z_log_var - K.square(z_mean) - K.exp(z_log_var), axis=-1)return xent_loss + kl_lossvae = Model(x, x_decoded_mean)
vae.compile(optimizer='rmsprop', loss=vae_loss)(x_train, y_train), (x_test, y_test) = fashion_mnist.load_data()x_train = x_train.astype('float32') / 255.
x_test = x_test.astype('float32') / 255.
x_train = x_train.reshape((len(x_train), np.prod(x_train.shape[1:])))
x_test = x_test.reshape((len(x_test), np.prod(x_test.shape[1:])))vae.fit(x_train, x_train,shuffle=True,epochs=epochs,batch_size=batch_size,validation_data=(x_test, x_test))encoder = Model(x, z_mean)x_test_encoded = encoder.predict(x_test, batch_size=batch_size)
plt.figure(figsize=(6, 6))
plt.scatter(x_test_encoded[:, 0], x_test_encoded[:, 1], c=y_test)
plt.colorbar()
plt.show()decoder_input = Input(shape=(latent_dim,))
_h_decoded = decoder_h(decoder_input)
_x_decoded_mean = decoder_mean(_h_decoded)
generator = Model(decoder_input, _x_decoded_mean)n = 20
digit_size = 28
figure = np.zeros((digit_size * n, digit_size * n))
grid_x = np.linspace(-3, 3, n)
grid_y = np.linspace(-3, 3, n)for i, xi in enumerate(grid_x):for j, yi in enumerate(grid_y):z_sample = np.array([[yi, xi]])x_decoded = generator.predict(z_sample)digit = x_decoded[0].reshape(digit_size, digit_size)figure[(n - i - 1) * digit_size: (n - i) * digit_size,j * digit_size: (j + 1) * digit_size] = digitplt.figure(figsize=(10, 10))
plt.imshow(figure)
plt.show()

我们来看一下隐层的表示,同样起到了很好的分类效果

然后再来生成一些图形,可以看到不同种类衣服之间的过渡

参考

  • Auto-Encoding Variational Bayes:https://arxiv.org/pdf/1312.6114.pdf
  • Tutorial on Variational Autoencoders:https://arxiv.org/pdf/1606.05908.pdf
  • Building Autoencoders in Keras:https://blog.keras.io/building-autoencoders-in-keras.html
  • Fashion-MNIST database of fashion articles:https://keras.io/datasets/#fashion-mnist-database-of-fashion-articles
  • 【啄米日常】变分编码器VAE:https://zhuanlan.zhihu.com/p/25269592

视频讲解课程

深度有趣(一)

转载于:https://my.oschina.net/u/3027864/blog/2088971

深度有趣 | 06 变分自编码器相关推荐

  1. 深度学习《变分自编码器》

    一个模型总是从简单到复杂,由粗糙到精致不断演化. 本文学习自:李宏毅机器学习视频(https://www.bilibili.com/video/av10590361/?p=29) 很多截图也都是来自于 ...

  2. 变分自编码器-推断输入分布的有向概率模型

    变分自编码器-推断输入分布的有向概率模型 0.摘要 本文主要分析一种用于生成新样本数据的有向概率模型--变分自编码器.总结相关的数学知识和方法,并对模型进行详细的分析与推导.用PyTorch深度学习框 ...

  3. 4.keras实现--生成式深度学习之用变分自编码器VAE生成图像(mnist数据集和名人头像数据集)...

    1.VAE和GAN 变分自编码器(VAE,variatinal autoencoder)   VS    生成式对抗网络(GAN,generative adversarial network) 两者不 ...

  4. 【阿里云课程】深度生成模型基础,自编码器与变分自编码器

    大家好,继续更新有三AI与阿里天池联合推出的深度学习系列课程,本次更新内容为第11课中两节,介绍如下: 第1节:生成模型基础 本次课程是阿里天池联合有三AI推出的深度学习系列课程第11期,深度生成模型 ...

  5. 深度学习之自编码器(4)变分自编码器

    深度学习之自编码器(4)变分自编码器 1. VAE原理  基本的自编码器本质上是学习输入 x\boldsymbol xx和隐藏变量 z\boldsymbol zz之间映射关系,它是一个 判别模型(Di ...

  6. 2020-4-22 深度学习笔记20 - 深度生成模型 5 (有向生成网络--sigmoid信念网络/可微生成器网络/变分自编码器VAE/生产对抗网络GAN/生成矩匹配网络)

    第二十章 深度生成模型 Deep Generative Models 中文 英文 2020-4-17 深度学习笔记20 - 深度生成模型 1 (玻尔兹曼机,受限玻尔兹曼机RBM) 2020-4-18 ...

  7. 【深度学习】用变分自编码器生成图像和生成式对抗网络

    目录 问题描述: 代码展示: VAE代码段 GAN部分(仅供参考) 运行截图: 参考: 问题描述: 从图像的潜在空间中采样,并创建全新图像或编辑现有图像,这是目前最流行也是最成 功的创造性人工智能应用 ...

  8. 深入理解自编码器(用变分自编码器生成图像)

    文章目录 自编码器 欠完备自编码器 正则自编码器 稀疏自编码器 去噪自编码器 收缩自编码器 变分自编码器 References 内容总结自花书<Deep Learning>以及<Py ...

  9. 【生成模型】变分自编码器(VAE)及图变分自编码器(VGAE)

    这段时间在学习机器学习中有关不确定性和概率分布的知识,发现了VAE这样一个有趣的方向,想抓紧时间整理一下VAE的主要思想和方法,然后思考如何迁移应用到自己的研究方向上. 从直观上理解VAE 变分自编码 ...

最新文章

  1. LeetCode第九题—— Palindrome Number(判断回文数)
  2. Judge Judy
  3. Windows 操作系统的安全设置
  4. Redis-01Redis概述
  5. web前端入门学习 css(10)移动端布局(学到DPG格式图片与webp格式图片停了)
  6. qemu-launcher:图形化的QEMU启动器
  7. python2处理耗时任务_RabbitMQ Go客户端教程2——任务队列/工作队列
  8. 堆和栈、值类型与引用类型、装箱与拆箱
  9. 2018年12月精选文章目录一览
  10. java分页插件PageHelper的内置list数据操作失败
  11. 【报告分享】产业互联网发展趋势及机会分析报告.pptx(附下载链接)
  12. Prepared statement needs to be re-prepare
  13. JAVA_返回一个数值的相反数的几种方式.
  14. SQL那些事儿(一)
  15. python在线编程免费课程-Python少儿基础编程课程
  16. Linux的版本可分为,Linux不同分类标准下的各种版本解读
  17. java声音变音代码实现_大话音频变声原理 附简单示例代码
  18. 《Arduino与LabVIEW开发实战》第3章 如何连接Arduino与LabVIEW
  19. 中职升高职c语言程序设计教程课后答案,锦职业技术学院2020年单独招生计算机应用技术专业技能测试考试大纲(中职升高职)...
  20. 国内物联网平台(7):Ablecloud物联网自助开发和大数据云平台

热门文章

  1. 解决AI大难题:如何降低AI运行对能源的消耗?
  2. 全球数字孪生市场大预测:2025 年的 358 亿美元,年复合增长率(CAGR)高达 37.8%...
  3. 德国再出颠覆性发明,这次要安排我们的快递
  4. 2018全球硬科技创新暨“一带一路”创新合作大会,这些硬科技都要来!
  5. 中国自动化学会平行智能专业委员会成立
  6. 周鸿祎:quot;安全大脑quot;将成智能经济时代的网络安全中枢
  7. 图解全球无人驾驶产业链:这些公司在主宰人类出行的未来
  8. 就很突然,电脑里多出了一个文件......
  9. 甲方乙方和站在外包中间的你 | 每日趣闻
  10. 漫画 | TCP,一个悲伤的故事