In my previous post I covered the theory behind Variational Autoencoders. It’s time now to get our hands dirty and develop some code that can lead us to a better comprehension of this technique. I decided to use Tensorflow since I want to improve my skills with it and adapt to the last changes that are being pushed towards the 2.0 version. Let’s code!
Note: All code in here can be found on my Github account.
You can read the other posts in this series here:
- Post 1: The theory behind Variational Autoencoders
- Post 3: Generating FIFA 19 players with VAEs and Tensorflow
Get the data
Tensorflow (with the recently incorporated Keras API) provides a reasonable amount of image datasets that we can use to test the performance of our network. It is super simple to import them without loosing time on data preprocessing.
Let’s start with the classics and import the MNIST dataset. For this, I will use another recently added API, the tf.dataset, which allows you to build complex input pipelines from simple, reusable pieces.
import tensorflow as tf | |
from tensorflow import keras | |
# Define parameters | |
batch_size = 128 | |
# Get data | |
fashion_mnist = keras.datasets.mnist | |
(train_images, train_labels), (test_images, test_labels) = fashion_mnist.load_data() | |
# Create tf dataset | |
with tf.variable_scope("DataPipe"): | |
dataset = tf.data.Dataset.from_tensor_slices(train_images) | |
dataset = dataset.map(lambda x: tf.image.convert_image_dtype([x], dtype=tf.float32)) | |
dataset = dataset.batch(batch_size=batch_size).prefetch(batch_size) | |
iterator = dataset.make_initializable_iterator() | |
input_batch = iterator.get_next() | |
input_batch = tf.reshape(input_batch, shape=[-1, 28, 28, 1]) |
In here we download the dataset using Keras. After this we create tensors that will be filled with the Numpy vectors of the training set. The map function will convert the image from uint8
to floats
. The dataset API allows us to define a batch size and to prefetch this batches, it uses a thread and an internal buffer to prefetch elements from the input dataset ahead of the time they are requested. Last three lines are used to create an iterator, so we can move through the different batches of our data and reshape them. Pretty simple!
If this is the first time you use this API I recommend you to test how is this working with this simple example that shows the first image of each batch.
import matplotlib.pyplot as plt | |
init_vars = [tf.local_variables_initializer(), tf.global_variables_initializer()] | |
with tf.Session() as sess: | |
sess.run([init_vars, iterator.initializer]) # Initialize variables and the iterator | |
while 1: # Iterate until we get out of range error! | |
try: | |
batch = sess.run(input_batch) | |
print(batch.shape) # Get batch dimensions | |
plt.imshow(batch[0,:,:,0] , cmap='gray') | |
plt.show() | |
except tf.errors.OutOfRangeError: # This exception is triggered when all batches are iterated | |
print('All batches have been iterated!') | |
break |
Easy isn’t it? Alright, let’s keep moving.
Encoder
We must now “code our encoder”. Since we are dealing with images, we are going to use some convolutional layers which will help us to maintain the spatial relations between pixels. I got the some of the ideas of how to structure the network from this great Felix Mohr’s post
My intention is to gradually reduce the dimensionality of our input. The images are 28x28
pixels, so if we add a convolutional layer with a stride of 2 and some extra padding too, we can reduce the image dimension to the half (review some CNN theory here). With two of these layers concatenated the final dimensions will be 7x7
(x64
if have into account the number of filters I am applying). After this I use another convolutional layer with stride 1 and ‘same’ padding which will maintain the vector size through the conv operations.
The mean and standard deviation of our Gaussian will be computed through two dense layers. We must note that the standard deviation of a Normal distribution is always positive, so I added a softplus function that will take care of this restriction (other works also apply a diagonal constraint on these weights). The epsilon value will be sampled from an unit normal distribution and then we will obtain our
def encoder(X): | |
activation = tf.nn.relu | |
with tf.variable_scope("Encoder"): | |
x = tf.layers.conv2d(X, filters=64, kernel_size=4, strides=2, padding='same', activation=activation) | |
x = tf.layers.conv2d(x, filters=64, kernel_size=4, strides=2, padding='same', activation=activation) | |
x = tf.layers.conv2d(x, filters=64, kernel_size=4, strides=1, padding='same', activation=activation) | |
x = tf.layers.flatten(x) | |
# Local latent variables | |
mean_ = tf.layers.dense(x, units=FLAGS.latent_dim, name='mean') | |
std_dev = tf.nn.softplus(tf.layers.dense(x, units=FLAGS.latent_dim), name='std_dev') # softplus to force >0 | |
# Reparametrization trick | |
epsilon = tf.random_normal(tf.stack([tf.shape(x)[0], FLAGS.latent_dim]), name='epsilon') | |
z = mean_ + tf.multiply(epsilon, std_dev) | |
return z, mean_, std_dev |
Tip: we can see how our network looks using TensorBoard! (full code on Github)
Decoder
When we speak about decoding images and neural networks, we must have a word in our mind, transpose convolutions! They work as an upsampling method with learning parameters, so they will be in charge of recovering the original image dimension from the latent variables one. It’s common to apply some non linear transformations using dense layers before the transposed ones. Below we can observe how the decoder is defined.
def decoder(z): | |
activation = tf.nn.relu | |
with tf.variable_scope("Decoder"): | |
x = tf.layers.dense(z, units=FLAGS.inputs_decoder, activation=activation) | |
x = tf.layers.dense(x, units=FLAGS.inputs_decoder, activation=activation) | |
recovered_size = int(np.sqrt(FLAGS.inputs_decoder)) | |
x = tf.reshape(x, [-1, recovered_size, recovered_size, 1]) | |
x = tf.layers.conv2d_transpose(x, filters=64, kernel_size=4, strides=1, padding='same', activation=activation) | |
x = tf.layers.conv2d_transpose(x, filters=64, kernel_size=4, strides=1, padding='same', activation=activation) | |
x = tf.layers.conv2d_transpose(x, filters=64, kernel_size=4, strides=1, padding='same', activation=activation) | |
x = tf.contrib.layers.flatten(x) | |
x = tf.layers.dense(x, units=28 * 28, activation=None) | |
x = tf.layers.dense(x, units=28 * 28, activation=tf.nn.sigmoid) | |
img = tf.reshape(x, shape=[-1, 28, 28, 1]) | |
return img |
Note that FLAGS.inputs_decoder
references the number of neurons in our dense layers which will determine the size of the image before the transpose convolution layers. In this case I will use the same structure as in the encoder, so the number of neurons will be 49 and the input image size to the first tranposed conv layer will be 7x7
. It’s also interesting to point that the last layer uses a sigmoid activation, so we force our outputs to be between 0 and 1.
Cost function
We have defined out encoder and decoder network but we still need to go trough a final step which consists on joining them and defining the cost function that will be optimized during the training phase.
# Link encoder and decoder | |
z, mean_, std_dev = encoder(input_batch) | |
output = decoder(z) | |
# Reshape input and output to flat vectors | |
flat_output = tf.reshape(output, [-1, 28 * 28]) | |
flat_input = tf.reshape(input_batch, [-1, 28 * 28]) | |
with tf.name_scope('loss'): | |
img_loss = tf.reduce_sum(flat_input * -tf.log(flat_output) + (1 - flat_input) * -tf.log(1 - flat_output), 1) | |
latent_loss = 0.5 * tf.reduce_sum(tf.square(mean_) + tf.square(std_dev) - tf.log(tf.square(std_dev)) - 1, 1) | |
loss = tf.reduce_mean(img_loss + latent_loss) |
I first connected the two networks and then (for simplicity sake) I have reshaped input and output batches of images to a flat vector. After this the image loss is computed using the binary cross entropy formula, which can be interpreted as the minus log likelihood of our data (more in here). We move on and compute the latent loss using the KL divergence formula (for a Gaussian and a unit Gaussian) and finally, we get the mean of all the image losses.
That’s all! We have created a complete network with its corresponding data input pipeline with just a few lines of code. We can move now, test our network and check if it can properly learn and create new amazing images!
Experiments
1. Are our networks learning?
Let’s start with a simple example and check that the network is working as it should. For this we will use the MNIST and fashion MNIST datasets and see how is the network reconstructing our input images after a few epochs. For this I will set the number of latent dimensions equal to 5.
Below, we can see how our network learns to transform the input image in the latent space of 5 dimensions, to recover them then to the original space. Note how as more epochs have passed, we obtain better results and a better loss.
We see how our network keeps improving the quality of the recovered images. It’s interesting to see how at the initial epochs numbers 6, 3, 4 are converted into something that we can considered as “similar” to a 9. But, after a few iterations the original input shape is conserved. For the fashion dataset we get a similar behavior with the bag pictures that are initially recovered as a boot!
It’s also curious, to observe how the images change to get an intuition of what is happening during the learning. But what is going on with the image encodings?
2. How does our latent space look?
In the previous section we used a 5 dimensional latent space but we can reduce this number to a two dimensional space which we can plot. In this way the complete images will be encoded in a 2D vector. Using a scatter plot, we can see how this dimensional space evolves with the number of epochs.
In a first moment all images are close to the prior (all point are located around 0). But during training the encoder learns to approximate the posterior distribution, so it will locate the latent variables in different parts of the space having into account their labels (equal label -> close region in the space). Let’s have a look first and then discuss more about this!
Isn’t this cool? Numbers that are similar are placed in a similar region of the space. For example we can see that zeros (red) and ones (orange) are easily recognized and located in a specific region of the space. Nevertheless it seems the network can’t do the same thing with eights (purple) and threes (green). Both of them occupy a very similar region.
Same thing happens with clothes. Similar garments as shirts, t-shirt and even dresses (which can be seen as an intersection between shirts and trousers) will be located in similar regions and the same thing happens with boots, sneakers and sandals!
These two dimensional plots can also help us to understand the KL divergence term. In my previous post I explained where does it come from and also that acts as a penalization (or regularization term) over the encoder when this one outputs probabilities that are not following an unit norm. Without this term the encoder can use the whole space and place equal labels in very different regions. Imagine for example two images of a number 1. Instead of being close one to each other in the space, the encoder could place them far away. This would result in a problem to generate unseen samples, since the space will be large and it will have ‘holes’ of information, empty areas that do not correspond to any number and are related with noise.
Look the x and y axis. On the left plot, there is not regularization, so points embrace a much larger region of the space, while as in the right image they are more concentrated, so this produces a dense space.
3. Generating samples
We can generate random samples that belong to our latent space. These points have not been used during training (they would correspond with a white space in previous plots). Our network decoder though, has learnt to reconstruct valid images that are related with those points without seem them. So let’s create a grid of points as the following one:
Each of this points can be passed to the decoder which will return us a valid images. With a few lines we can check how is our encoder doing through the training and evaluate the quality of the results. Ideally, all labels would be represented in our grid.
# Create mesh grid of values | |
values = np.arange(-3, 4, .5) | |
xx, yy = np.meshgrid(values, values) | |
input_holder = np.zeros((1, 2)) | |
# Matrix that will contain the grid of images | |
container = np.zeros((28 * len(values), 28 * len(values))) | |
for row in range(xx.shape[0]): | |
for col in range(xx.shape[1]): | |
input_holder[0, :] = [xx[row, col], yy[row, col]] | |
artificial_image = sess.run(output, feed_dict={z: input_holder}) | |
container[row * 28: (row + 1) * 28, col * 28: (col + 1) * 28] = np.squeeze(artificial_image) | |
plt.imshow(container, cmap='gray') | |
plt.savefig(os.path.join(results_folder, 'Test/Space_{}'.format(epoch))) | |
plt.close() |
What about the fashion dataset? Results are even more fun! Look how the different garment are positioned by ‘similarity’ in the space. Also the grid generated images look super real!
After 50 epochs of training and using the grid technique and the fashion MNIST dataset we achieve these results:
All this images here are fake. We can finally see how our encoder works and how our latent space has been able to properly encode 2D image representations. Observe how you can start with a sandal and interpolate points until you get a sneaker or even a boot!
Conclusion
We have learnt about Variational Autoencoders. We started with the theory and main assumptions that lie behind them and finally we implement this network using Google’s Tensorflow.
-
We have used the dataset API, reading and transforming the data to train our model using an iterator.
-
We have also implemented our encoder and decoder networks in a few lines of code.
-
The cost function, made up of two different parts, the log likelihood and the KL divergence term (which acts as a regularization term), should be also clear now.
-
The experimental section was designed to support all the facts mentioned above so we could see with real images what is going on under all the computations.
It could be interesting to adapt our network to work with larger and colored images (3 channels) and observe how our network does with a dataset like that. I might implement some of these features but this is all for now!
Any ideas for future posts or is there something you would like to comment? Please feel free to reach out via Twitter or Github