Beginner’s Guide to Building Neural Networks in TensorFlowDetailed Walkthrough of the TensorFlow 2.
0 Beginner NotebookEvan HeitmanBlockedUnblockFollowFollowingJun 5If you’re reading this you’ve probably had some exposure to neural networks and TensorFlow, but you might feel somewhat daunted by the various terms associated with deep learning that are often glossed over or left unexplained in many introductions to the technology.
This article will shine a light on some of these topics by doing an in-depth walkthrough of TensorFlow 2.
0’s beginner tutorial.
This guide is based on the TensorFlow 2.
0 beginner notebook and I recommend taking a look at it and running it in Google Colab (it’s only 16 lines of code!) to maximize your comprehension of the material covered here.
NOTE: Given that this is a beginner’s guide, there should be a low barrier-to-entry to most of the ideas expressed herein, but some some background familiarity with with what neural networks are will be helpful.
This article provides a good overview in case you want to brush up.
What You’ll LearnAfter reading this you’ll have a better understanding of some key conceptual topics and TensorFlow/Keras implementation of those topics (Keras is a deep learning library built on top of TensorFlow).
Conceptual:Neural network layer shapesActivation functions (such as Relu and Softmax)LogitsDropoutOptimizersLossEpochsTensorFlow/Keras functions:tf.
fit()The DataThe data that the TensorFlow 2.
0 beginner tutorial uses is the MNIST dataset which is considered a kind of “Hello, World!” for neural networks and deep learning, and it can be downloaded directly from Keras.
It is a dataset full of hand-drawn digits ranging from 0–9 with a corresponding label describing what digit the drawing is supposed to be depicting.
Example observations from MNIST dataset The idea behind working with this dataset is that we want to be able to train a model that learns what kinds of shapes correspond to digits 0–9 and that subsequently is able to correctly label images that it hasn’t trained on.
This task becomes more complicated when images such as the one pictured below are passed to the model.
Some humans might even mistake this drawing for a zero even though it’s labelled as an eight.
Example an ambiguous image from MNIST (index 3570 of training set)On a high level, the model built in the beginner tutorial takes in training images as inputs and tries to classify those images as a digit from 0–9.
If it makes an incorrect prediction, it will make mathematical adjustments to better predict similar images.
Once the model has finished training, it will be tested on images that it hasn’t trained on to get a final assessment of how well the model performs.
Basic structure of the neural net built in the beginner notebookUnderstanding the Beginner NotebookNow let’s take a deep dive into how TensorFlow implements this basic neural network.
Loading the dataAfter doing some set up in the first cell, the notebook starts by loading the MNIST dataset from the Keras library using its load_data() function which returns two tuples as shown in the code.
Documentation can be found here.
mnist = tf.
mnist(x_train, y_train), (x_test, y_test) = mnist.
load_data()It will be helpful to understand what this data actually looks like as we go into exploring what TensorFlow does with it.
shape(60000, 28, 28)>>> y_train.
shape(10000, 28, 28)>>> y_test.
shape(10000,)Looking that these results we can see that there are 70k total images in the dataset, 60k training and 10k testing.
The two 28’s indicate that each image is 28 pixels by 28 pixels and images are represented as 28×28 arrays filled with pixel values as can seen in the following visualization.
Depiction of how each image is stored in the MNIST datasetThe last step that the notebook performs to prepare the data is converting every pixel value in each image to a floating-point number between 0.
This is done to help with the scale in the math involved in producing a prediction for each image.
x_train, x_test = x_train / 255.
0, x_test / 255.
0Building the model structurePerhaps the most confusing part of this notebook is the part where the model structure is created.
model = tf.
Flatten(input_shape=(28, 28)), tf.
Dense(128, activation='relu'), tf.
Dense(10, activation='softmax')])The purpose of this code is to specify what kind of layers are going to be present in our neural net.
The first component of this is the tf.
All this function does is begin the creation of a linear (or “sequential”) arrangement of layers.
All the other code in the above snippet detail which layers will be in the model and how they will be arranged.
The next line of code tf.
Flatten(input_shape=(28,28)) creates the first layer in our network.
Intuitively, we want to be able to use all of the information in an image to predict what digit it is, and so the input layer should have a node for each pixel in an image.
Each image has 28*28 =784 values and so Flatten() creates a layer with 784 nodes that contain each pixel value for a given image.
If we had color images that contain 3 values for each pixel (RGB values) then Flatten() would create a layer with 28*28*3 = 2352 nodes.
The other kind of layer we see in the model is created using tf.
Dense() which creates what is called a fully-connected or densely-connected layer.
This can be compared to a sparsely-connected layer and the distinction has to do with how information is passed between nodes in adjacent layers.
Comparison between dense and sparse connectivity (image from Amir Alavi’s blog)You can see that in a densely-connected layers, each node in one layer is connected to each node in the next layer, whereas in sparsely-connected layers this is not the case.
And so what Dense() does is create a layer that is fully connected to the layer that precedes it.
The first parameter (128 in the first instance) specifies how many nodes should be in the layer.
The number of nodes in hidden layers (layers that aren’t the input or output layers) is somewhat arbitrary but an important thing to note is that the output layer has a number of nodes equal to how many classes the model is trying to predict.
In this case, the model is trying to predict between 10 different digits and so the final layer in the model has 10 nodes.
This is crucial because the output for each node of the final layer will be a probability that a given image is a particular digit.
In order to understand the rest of code in this snippet we need to develop an understanding of activation functions and dropout.
Activation FunctionsAs important as the layout and structure of a neural net is, it’s good to remember that, at the end of the day, what a neural net does is a lot of math.
Each node takes the values of the nodes in the previous layer and computes a weighted sum of them, producing a scalar value which is called a logit.
Much like how a neuron in the human brain “fires” when prompted by certain inputs, we must specify how each node (which are also sometimes referred to as neurons) in our network “fires” when it is given certain inputs.
This is what activation functions do.
They take logits, the results of the aforementioned weighted sum, and convert them to an “activation” based on what function is being used.
A common activation function, and the one used in the first Dense() layer in our network, is called “ReLU” which is short for rectified linear unit.
ReLU activation functionWhat ReLU does is make the activation of any negative logits 0 (the node does not fire), while leaving any positive logits unchanged (the node fires with strength linearly proportional to the strength of the input).
For more information on the power of ReLU and why it is useful check out this article.
Another common activation function, and the one that is used in the second instance of Dense(), is called “softmax”.
Udacity Deep Learning slide on softmaxAs the image above shows, softmax takes the logits computed by the weighted sum of the activations from the previous layer and converts them to probabilities that sum to 1.
This makes it an extremely useful activation function to use in our output layer because it provides easy-to-interpret results for the likelihood of an image being a particular digit.
There are many other activation functions and deciding which one to use is often a matter of experimentation or heuristic judgement (or both).
A good overview of some other activation functions can be found here (I recommend reading it if you’re having trouble understanding why activation functions are necessary in the first place).
DropoutThe last unexplained piece of the code snippet we’ve been examining so far is the call of tf.
The concept of dropout goes back to the earlier discussion of the connectivity of layers, and has to do specifically with a few drawbacks associated with densely-connected layers.
One of the drawbacks of densely-connected layers is that it can lead to very computationally-expensive neural nets.
With each node transmitting information to every other node in the next layer, the complexity of the weighted sums computed in each node increases exponentially with the number of nodes in each layer.
Another drawback is that with so much information being passed from layer to layer, models can have a tendency to overfit to the training data, ultimately hurting performance.
This is where dropout comes in.
Dropout makes it so some of the nodes in a given layer don’t pass on their information to the next layer.
This helps with computation time and with overfitting.
So in the beginner notebook, the call of Dropout(0.
2) in between the two Dense() layers makes it so that each node in the first Dense() layer has a 0.
2 probability of being dropped from the computation of the activations of the next layer.
You might have caught on that this effectively makes the output layer in the model a sparsely-connected layer.
Srivastava, Nitish, et al.
”Dropout: a simple way to prevent neural networks fromoverfitting”, JMLR 2014Now that we understand all the components of our model, let’s take advantage of the model.
summary() function and do a sanity check on what our network structure looks like.
After double checking the output shapes, it all looks good, so now let’s move on to compiling, training, and running the model!Compiling, Training, and Running the Neural NetNow that we’ve specified what our neural net looks like, the next step is to tell Tensorflow how to train it.
Compiling the modelThe snippet of code we’ll be examining in this section is the following:model.
compile(optimizer='adam', loss='sparse_categorical_crossentropy', metrics=['accuracy'])The model.
compile()function is called on a pre-built model and it specifies the loss function, optimizer, and metrics, each of which will be explained.
These are important features of how a neural network produces its final predictions.
Loss FunctionAt the beginning of this guide, it was mentioned that on a high-level, the model built in the beginner notebook will learn how to classify certain images as digits and it does this by making a prediction, seeing how far off its prediction was from the correct answer, and then updating itself to better predict those kinds of digits.
The loss function is the part of the model that quantifies how far off a prediction is from the correct answer.
Different kinds of models are going to need different kinds of loss functions.
For example, the loss function for a problem like this one where the outputs to our model are probabilities would have to very different from the loss function of a model that is trying to predict something like price in dollars.
The loss function for this particular model is ‘sparse_categorical_crossentropy’ which is good for multiclass-classification problems like this one.
In our case, if the model predicts that an image has only a small probability of being its actual label, that will lead to a high loss.
OptimizerAnother way of expressing what training a model actually means is that it seeks to minimize the loss.
If loss is a measurement of how far off a prediction is from the correct answer, and a higher loss means a more incorrect prediction, seeking to minimize the loss is a quantifiable way of determining how well a model is performing.
As was mentioned earlier, a key part of training neural nets is revising the mathematics parameters of the nodes of a network based off of how effective those parameters were in classifying an image.
In a process called backpropagation, neural nets use a mathematical tool called gradient descent to update parameters to improve the model.
The details of those terms are somewhat outside of the scope of this guide but for the purposes of understanding what the beginner notebook is doing, the optimizer parameter of the model.
compile() function specifies a way for making the backpropagation process more faster and more effective.
The “adam” optimizer is a commonly used optimizer and works fine for this problem.
MetricsThe last part of the model.
compile() function is specifying the metrics it should use in evaluating the model.
Accuracy is a useful, but imperfect metric, for gauging model performance and it should be used by itself with a some amount of caution (for more on why that is and some other potential metrics, you can take a look at this article).
Training the modelAt long last comes actually training the model, and with TensorFlow 2.
0, this is very easy to do.
fit(x_train, y_train, epochs=5)This line of code is pretty intuitive, passing the training data and the correct labels of that data.
The epoch parameter in the model.
fit() function is the number of times the model sees all of the training data.
The reason why we want the model to see all of the training data multiple times is that one go-through might not be enough for the model to sufficiently update its weights when computing weighted sums to make a marked improvement in predictive power.
Results from training the model with 5 epochsRunning this code, you can see that in each epoch, the model goes through all 60k images in the training set.
You can also see the the loss decreases and the accuracy increases with each epoch, meaning that the model is getting better and better at classifying digits with each epoch.
Evaluating the modelAnd finally, using model.
evaluate(x_test, y_test), we can predict the classes of our test set and see how our model performs.
This plot shows that the even though training accuracy keeps going up the more epochs we train, the validation accuracy is starting to plateau or decrease which indicates that we probably don’t need to train for more than 5 epochs.
Confusion matrix for our neural netFor a more detailed evaluation of how the model performed, we can build a confusion matrix.
From this confusion matrix we can see that our model has the most trouble with nines and tends to confuse them with sevens, fours, or threes.
ConclusionCongratulations!.You’ve made it through this guide to TensorFlow 2.
0’s beginner notebook and now have a better understanding of the shapes of neural network layers, activation functions, logits, dropout, optimizers, loss functions and loss, and epochs.
You also gained familiarity with how to implement these concepts using TensorFlow/Keras!.For more practice, I recommend experimenting with the different parameters discussed in this guide to see what impact they have on model performance.
Happy modeling!.. More details