6

生成对抗网络简介 (GAN Introduction)

 2 years ago
source link: https://leovan.me/cn/2018/02/gan-introduction/
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.
neoserver,ios ssh client

生成对抗网络简介 (GAN Introduction)

范叶亮 / 2018-02-03

分类: 机器学习, 深度学习 / 标签: GAN, DCGAN, 生成对抗网络 / 字数: 3962


Generative Adversarial Networks (GAN)

生成对抗网络 (Generative Adversarial Network, GAN) 是由 Goodfellow 1 于 2014 年提出的一种对抗网络。这个网络框架包含两个部分,一个生成模型 (generative model) 和一个判别模型 (discriminative model)。其中,生成模型可以理解为一个伪造者,试图通过构造假的数据骗过判别模型的甄别;判别模型可以理解为一个警察,尽可能甄别数据是来自于真实样本还是伪造者构造的假数据。两个模型都通过不断的学习提高自己的能力,即生成模型希望生成更真的假数据骗过判别模型,而判别模型希望能学习如何更准确的识别生成模型的假数据。

GAN 由两部分构成,一个生成器 (Generator) 和一个判别器 (Discriminator)。对于生成器,我们需要学习关于数据 x 的一个分布 pg,首先定义一个输入数据的先验分布 pz(z),其次定义一个映射 G(z;θg):z→x。对于判别器,我们则需要定义一个映射 D(x;θd) 用于表示数据 x 是来自于真实数据,还是来自于 pg。GAN 的网络框架如下图所示 2

Goodfellow 在文献中给出了一个重要的公式用于求解最优的生成器

(1)minGmaxDV(D,G)=Ex∼pdata(x)[log⁡D(x)]+Ez∼pz(z)[log⁡(1−D(G(z)))]

上式中,在给定的 G 的情况下,maxDV(G,D)衡量的是 pdata 和 pg 之间的“区别”,因此我们最终的优化目标就是找到最优的 G∗ 使得 pdata 和 pg 之间的“区别”最小。

首先,在给定 G 的时候,我们可以通过最大化 V(G,D) 得到最优 D∗

(2)V(G,D)=Ex∼pdata(x)[log⁡D(x)]+Ez∼pz(z)[log⁡(1−D(G(z)))]=∫xpdata(x)log⁡D(x)dx+∫zpz(z)log⁡(1−D(g(z)))dz=∫xpdata(x)log⁡D(x)+pg(x)log⁡(1−D(x))dx

对于给定的任意 a,b∈R2∖{0,0},alog⁡(x)+blog⁡(1−x)在 x=aa+b 处取得最大值,D 的最优值为

(3)DG∗=pdata(x)pdata(x)+pg(x)

因此,maxDV(G,D) 可重写为

(4)C(G)=maxDV(G,D)=V(G,D∗)=Ex∼pdata(x)[log⁡DG∗(x)]+Ez∼pz(z)[log⁡(1−DG∗(G(z)))]=Ex∼pdata(x)[log⁡DG∗(x)]+Ex∼pg(x)[log⁡(1−DG∗(x))]=Ex∼pdata(x)[log⁡pdata(x)pdata(x)+pg(x)]+Ex∼pg(x)[log⁡pg(x)pdata(x)+pg(x)]=∫xpdata(x)log⁡12pdata(x)pdata(x)+pg(x)2dx+∫xpg(x)log⁡12pg(x)pdata(x)+pg(x)2dx=∫xpdata(x)log⁡pdata(x)pdata(x)+pg(x)2dx+∫xpg(x)log⁡pg(x)pdata(x)+pg(x)2dx+2log⁡12=KL(pdata(x)‖pdata(x)+pg(x)2)+KL(pg(x)‖pdata(x)+pg(x)2)−2log⁡2=2JS(pdata(x)‖pg(x))−2log⁡2

其中 KL 表示 KL 散度 3,JS 表示 JS 散度 4,因此在全局最优情况下 pg=pdata。

整个 GAN 的训练过程如下所示:

\begin{algorithm}
\caption{Minibatch SGD for GAN 算法}
\begin{algorithmic}
\REQUIRE $iter, k, m$
\ENSURE $\theta_d, \theta_g$
\FOR{$i = 1, 2, ..., iter$}
    \FOR{$j = 1, 2, ..., k$}
        \STATE Sample minibatch of $m$ noise samples $\{z^{\left(1\right)}, ..., z^{\left(m\right)}\}$ from $p_g \left(\boldsymbol{z}\right)$
        \STATE Sample minibatch of $m$ examples $\{x^{\left(1\right)}, ..., x^{\left(m\right)}\}$ from $p_{data} \left(\boldsymbol{z}\right)$
        \STATE $\theta_d \gets \theta_d \textcolor{red}{+} \nabla_{\theta_d} \dfrac{1}{m} \sum_{i=1}^{m}{\left[\log D \left(x^{\left(i\right)}\right) + \log \left(1 - D \left(G \left(z^{\left(i\right)}\right)\right)\right)\right]}$
    \ENDFOR
    \STATE Sample minibatch of $m$ noise samples $\{z^{\left(1\right)}, ..., z^{\left(m\right)}\}$ from $p_g \left(\boldsymbol{z}\right)$
    \STATE $\theta_g \gets \theta_g \textcolor{red}{-} \nabla_{\theta_g} \dfrac{1}{m} \sum_{i=1}^{m}{\log \left(1 - D \left(G \left(z^{\left(i\right)}\right)\right)\right)}$
\ENDFOR
\end{algorithmic}
\end{algorithm}

Algorithm 1 Minibatch SGD for GAN 算法

Require: iter,k,m

Ensure: θd,θg

1:for i=1,2,...,iter do

2:for j=1,2,...,k do

3:Sample minibatch of m noise samples {z(1),...,z(m)} from pg(z)

4:Sample minibatch of m examples {x(1),...,x(m)} from pdata(z)

5:θd←θd+∇θd1m∑i=1m[log⁡D(x(i))+log⁡(1−D(G(z(i))))]

6:end for

7:Sample minibatch of m noise samples {z(1),...,z(m)} from pg(z)

8:θg←θg−∇θg1m∑i=1mlog⁡(1−D(G(z(i))))

9:end for

在实际的训练过程中,我们通常不会直接训练 G 最小化 log⁡(1−D(G(z))),因为其在学习过程中的早起处于饱和状态,因此我们通常会通过最大化 log⁡(D(G(z)))。

存在的问题

针对 GAN,包括 Goodfellow 自己在内也提出了其中包含的很多问题 2,因此后人也提出了大量的改进,衍生出了大量的 GAN 变种。本章节仅对原始的 GAN 中存在的问题进行简略介绍,相关的改进请参见后续的具体改进算法。

JS 散度问题

我们在训练判别器的时候,其目标是最大化 JS 散度,但 JS 散度真的能够很好的帮助我们训练判别器吗? Wasserstein GAN 一文 5给出了不同生成器情况下 JS 散度的变化情况。

上图中,左边为一个基于 MLP 的生成器,右边为一个 DCGAN 6 生成器,两者均有一个 DCGAN 的判别器。根据上文我们可以知道判别器的目标是最大化

(5)L(D,θg)=Ex∼pdata(x)[log⁡DG∗(x)]+Ex∼pg(x)[log⁡(1−DG∗(x))]=2JS(pdata(x)‖pg(x))−2log⁡2

上图中 Y 轴绘制的为 12L(D,θg)+log⁡2,因为 −2log⁡2≤L(D,θg)≤0,因此我们可得 0≤12L(D,θg)+log⁡2≤log⁡2。从图中我们可以看出,针对两种不同的情况,其值均很快的逼近最大值 log⁡2≈0.69,当接近最大值的时候,判别器将具有接近于零的损失,此时我们可以发现,尽管 JS 散度很快趋于饱和,但 DCGAN 生成器的效果却仍在不断的变好,因此,使用 JS 散度作为判别其的目标就显得不是很合适。

多样性问题 Mode Collapse

Mode Collapse 问题是指生成器更多的是生成了大量相同模式的数据,导致的结果就是生成的数据缺乏多样性,如下图所示 7:

不难看出,其中红色方框圈出来的图像十分相似,这样的问题我们就称之为 Mode Collapse。Goolfellow 曾经从不同的 KL 散度的角度解释引起 Mode Collapse 的问题,但最后发现其并非由散度的不同所导致。对于 KL 散度,其并非是对称的,即 DKL(pdata‖pmodel) 与 DKL(pmodel‖pdata) 是不同的。在最大化似然估计的时候使用的是前者,而在最小化 JS 散度的时候使用的更类似于后者。如下图所示

假设我们的模型 q 并没有足够能能力去拟合真实数据分布 p,假设真实数据由两个二维的高斯分布构成,而模型需要使用一个一维的高斯分布去拟合。在左图中,模型更倾向于覆盖两个高斯分布,也就是说其更倾向与在有真实数据的地方得到更大的概率。在右图中,模型更倾向于覆盖其中一个高斯分布,也就是说其更倾向于在没有真实数据的地方取得更小的概率。这样,如果我们用 JS 散度训练模型的时候就容易出现模式缺失的问题,但尽管我们利用前者去优化模型,但结果中仍然出现了 Mode Collapse 的问题,这也就说明并非 JS 散度问题导致的 Mode Collapse。

针对 Mode Collapse 的问题,出现了大量不同角度的优化

  • 基于正则化的优化 8
  • 基于 Minibatch 的优化 9
  • 基于 Unrolled Optimization 的优化 10
  • 基于集成算法的优化 11

MNIST 示例

我们利用 MNIST 数据集测试原始的 GAN 模型的效果,代码主要参考了 Keras-GAN,最终实现代码详见 image_gan_keras.py,我们简单对其核心部分进行说明。

  • def build_generator(self):
      model = Sequential()
      
      model.add(Dense(int(self._hidden_dim / 4),
                          input_shape=self._noise_shape))
      model.add(LeakyReLU(alpha=0.2))
      model.add(BatchNormalization(momentum=0.8))
      model.add(Dense(int(self._hidden_dim / 2)))
      model.add(LeakyReLU(alpha=0.2))
      model.add(BatchNormalization(momentum=0.8))
      model.add(Dense(self._hidden_dim))
      model.add(LeakyReLU(alpha=0.2))
      model.add(BatchNormalization(momentum=0.8))
      model.add(Dense(np.prod(self._input_shape), activation='tanh'))
      model.add(Reshape(self._input_shape))
      
      print('Generator Summary: ')
      model.summary()
      
      noise = Input(shape=self._noise_shape)
      image = model(noise)
          
      return Model(noise, image)
    

在生成器中,我们使用了一个包含3个隐含层的全链接网络,其中 self._hidden_dim 是我们定义的隐含节点最多一层的节点数;self._noise_shape 为用于生成器的噪音数据的形状;self._input_shape 为输入数据形状,即图片数据的形状,中间层次采用的激活函数为 LeakyReLU,最后一层采用的激活函数为 tanh

  • def build_discriminator(self):
      model = Sequential()
      
      model.add(Flatten(input_shape=self._input_shape))
      model.add(Dense(int(self._hidden_dim / 2)))
      model.add(LeakyReLU(alpha=0.2))
      model.add(Dense(int(self._hidden_dim / 4)))
      model.add(LeakyReLU(alpha=0.2))
      model.add(Dense(1, activation='sigmoid'))
      
      print('Discriminator Summary: ')
      model.summary()
      
      image = Input(shape=self._input_shape)
      label = model(image)
          
      return Model(image, label)
    

在判别器中,我们使用了一个包含2个隐含层的全链接网络,中间层次采用的激活函数为 LeakyReLU,最后一层采用的激活函数为 sigmoid

  • class ImageBasicGAN():
      def __init__(self, width, height, channels,
                   a_optimizer=Adam(1e-4, beta_1=0.5),
                   g_optimizer=Adam(1e-4, beta_1=0.5),
                   d_optimizer=Adam(1e-4, beta_1=0.5),
                   noise_dim=100, hidden_dim=1024):
          '''
      
          Args:
              width: 图像宽度
              height: 图像高度
              channels: 图像颜色通道数
              a_optimizer: 对抗网络优化器
              g_optimizer: 生成器优化器
              d_optimizer: 判别器优化器
              noise_dim: 噪音数据维度
              hidden_dim: 隐含层最大维度
          '''
              
          # 省略一大坨代码
      
          # 构建和编译判别器
          self._discriminator = self.build_discriminator()
          self._discriminator.compile(loss='binary_crossentropy',
                                      optimizer=d_optimizer,
                                      metrics=['accuracy'])
      
          # 构建和编译生成器
          self._generator = self.build_generator()
          self._generator.compile(loss='binary_crossentropy',
                                  optimizer=g_optimizer)
      
          # 生成器利用噪声数据作为输入
          noise = Input(shape=self._noise_shape)
          generated_image = self._generator(noise)
      
          # 当训练整个对抗网络时,仅训练生成器
          self._discriminator.trainable = False
      
          # 判别器将生成的图像作为输入
          label = self._discriminator(generated_image)
      
          # 构建和编译整个对抗网络
          self._adversarial = Model(noise, label)
          self._adversarial.compile(loss='binary_crossentropy',
                                    optimizer=a_optimizer)
    

在构造整个对抗网络的时候,需要注意我们训练完判别器后,通过训练整个对抗网络进而训练生成器的时候是固定住训练好的判别器的,因此在训练整个对抗网络的时候我们应该将判别器置为无需训练的状态。

  • def train(self, x_train, output_dir, iters,
            batch_size=32, k=1, save_interval=200):
      ''' 训练模型
      
      Args:
          x_train: 训练数据
          output_dir: 相关输出路径
          iters: 迭代次数
          batch_size: 批大小
          k: K
          save_interval: 结果保存间隔
      '''
          
      # 省略一大坨代码
      
      for iter in range(iters):
          # 训练判别器
          for _ in range(k):
              train_indices = np.random.randint(0, x_train.shape[0],
                                                batch_size)
              train_images = x_train[train_indices]
      
              noises = np.random.normal(0, 1, (batch_size, self._noise_dim))
              generated_images = self._generator.predict(noises)
      
              self._discriminator.train_on_batch(train_images,
                                                 np.ones((batch_size, 1)))
              self._discriminator.train_on_batch(generated_images,
                                                 np.zeros((batch_size, 1)))
                  
          # 训练生成器
          noises = np.random.normal(0, 1, (batch_size, self._noise_dim))
          labels = np.ones(batch_size)
      
          self._adversarial.train_on_batch(noises, labels)
      
      # 再省略一大坨代码
    

在训练整个对抗网络的时候,我们对于一个给定的生成器,我们将生成器生成的数据作为负样本,将从真实数据中采样的数据作为正样本训练判别器。Goodfellow 在描述 GAN 训练的过程中,对于给定的生成器,训练判别器 k 次,不过通常取 k=1。训练好判别器后,再随机生成噪音数据用于训练生成器,周而复始直至达到最大迭代次数。

在整个训练过程中,我们分别记录了判别器和生成器的损失的变化,以及判别器的准确率的变化,如下图所示:

从上图中我们可以看出,在训练开始阶段,判别器能够相对容易的识别出哪些数据是来自于真实数据的采样,哪些数据是来自于生成器的伪造数据。随着训练的不断进行,判别器的准确率逐渐下降,并稳定在 60% 左右,也就是说生成器伪造的数据越来越像真实的数据,判别器越来越难进行甄别。

下图中我们展示了利用 MNIST 数据集,进行 30000 次的迭代,每 1000 次截取 100 张生成器利用相同噪音数据伪造的图像,最后合成的一张生成图片的变化动图。

Deep Convolutional GAN

DCGAN (Deep Convolutional GAN) 是由 Radford 6 等人提出的一种对原始 GAN 的变种,其基本的思想就是将原始 GAN 中的全链接层用卷积神经网络代替。在文中,Radford 等人给出构建一个稳定的 DCGAN 的建议,如下:

  • 在网络中不使用 pooling 层,而是使用多步长的卷积层 (判别器) 和多步长的反卷积层 (生成器)。
  • 在生成器和判别器中均使用批标准化。
  • 对于深层的框架,去掉全链接层。
  • 在生成器中使用 ReLU 激活函数,最后一层使用 Tanh 激活函数。
  • 在判别器中使用 LeakyReLU 激活函数。

我们利用 MNIST 数据集测试 DCGAN 模型的效果,最终实现代码详见 image_dcgan_keras.py。训练过程中判别器和生成器的损失的变化,以及判别器的准确率的变化,如下图所示:

下图中我们展示了利用 MNIST 数据集,进行 30000 次的迭代,每 1000 次截取 100 张生成器利用相同噪音数据伪造的图像,最后合成的一张生成图片的变化动图。

从生成的结果中可以看出,DCGAN 生成的图片的质量还是优于原始的 GAN 的,在原始的 GAN 中我们能够明显的看出其中仍旧包含大量的噪音点,而在 DCGAN 中这种情况几乎不存在了。

「真诚赞赏,手留余香」

  1. Goodfellow, Ian, et al. “Generative adversarial nets.” Advances in neural information processing systems. 2014.
  2. Goodfellow, Ian. “NIPS 2016 tutorial: Generative adversarial networks.” arXiv preprint arXiv:1701.00160 (2016).
  3. https://en.wikipedia.org/wiki/Kullback–Leibler_divergence
  4. https://en.wikipedia.org/wiki/Jensen–Shannon_divergence
  5. Arjovsky, Martin, Soumith Chintala, and Léon Bottou. “Wasserstein gan.” arXiv preprint arXiv:1701.07875 (2017).
  6. Radford, Alec, Luke Metz, and Soumith Chintala. “Unsupervised representation learning with deep convolutional generative adversarial networks.” arXiv preprint arXiv:1511.06434 (2015).
  7. http://speech.ee.ntu.edu.tw/~tlkagk/courses/MLDS_2017/Lecture/GAN%20(v11).pdf
  8. Che, Tong, et al. “Mode regularized generative adversarial networks.” arXiv preprint arXiv:1612.02136 (2016).
  9. Salimans, Tim, et al. “Improved techniques for training gans.” Advances in Neural Information Processing Systems. 2016.
  10. Metz, Luke, et al. “Unrolled generative adversarial networks.” arXiv preprint arXiv:1611.02163 (2016).
  11. Tolstikhin, Ilya O., et al. “Adagan: Boosting generative models.” Advances in Neural Information Processing Systems. 2017.
「真诚赞赏,手留余香」

Ising 模型,Hopfield 网络和受限的玻尔兹曼机 (Ising, Hopfield and RBM) 深度学习优化算法 (Optimization Methods for Deeplearning)


© 2017-2021 Leo Van
Github · ORCID · I am Mr. Black.

About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK