Author: Justin Johnson
Note
This is one of our older PyTorch tutorials. You can view our latest beginner content in Learn the Basics.
This tutorial introduces the fundamental concepts of PyTorch through self-contained examples.
At its core, PyTorch provides two main features:
- An n-dimensional Tensor, similar to numpy but can run on GPUs
- Automatic differentiation for building and training neural networks
We will use a problem of fitting y=\sin(x) with a third order polynomial as our running example. The network will have four parameters, and will be trained with gradient descent to fit random data by minimizing the Euclidean distance between the network output and the true output.
Note
You can browse the individual examples at the :ref:`end of this page <examples-download>`.
Table of Contents
Before introducing PyTorch, we will first implement the network using numpy.
Numpy provides an n-dimensional array object, and many functions for manipulating these arrays. Numpy is a generic framework for scientific computing; it does not know anything about computation graphs, or deep learning, or gradients. However we can easily use numpy to fit a third order polynomial to sine function by manually implementing the forward and backward passes through the network using numpy operations:
.. includenodoc:: /beginner/examples_tensor/polynomial_numpy.py
Numpy is a great framework, but it cannot utilize GPUs to accelerate its numerical computations. For modern deep neural networks, GPUs often provide speedups of 50x or greater, so unfortunately numpy won't be enough for modern deep learning.
Here we introduce the most fundamental PyTorch concept: the Tensor. A PyTorch Tensor is conceptually identical to a numpy array: a Tensor is an n-dimensional array, and PyTorch provides many functions for operating on these Tensors. Behind the scenes, Tensors can keep track of a computational graph and gradients, but they're also useful as a generic tool for scientific computing.
Also unlike numpy, PyTorch Tensors can utilize GPUs to accelerate their numeric computations. To run a PyTorch Tensor on GPU, you simply need to specify the correct device.
Here we use PyTorch Tensors to fit a third order polynomial to sine function. Like the numpy example above we need to manually implement the forward and backward passes through the network:
.. includenodoc:: /beginner/examples_tensor/polynomial_tensor.py
In the above examples, we had to manually implement both the forward and backward passes of our neural network. Manually implementing the backward pass is not a big deal for a small two-layer network, but can quickly get very hairy for large complex networks.
Thankfully, we can use automatic differentiation to automate the computation of backward passes in neural networks. The autograd package in PyTorch provides exactly this functionality. When using autograd, the forward pass of your network will define a computational graph; nodes in the graph will be Tensors, and edges will be functions that produce output Tensors from input Tensors. Backpropagating through this graph then allows you to easily compute gradients.
This sounds complicated, it's pretty simple to use in practice. Each Tensor
represents a node in a computational graph. If x
is a Tensor that has
x.requires_grad=True
then x.grad
is another Tensor holding the
gradient of x
with respect to some scalar value.
Here we use PyTorch Tensors and autograd to implement our fitting sine wave with third order polynomial example; now we no longer need to manually implement the backward pass through the network:
.. includenodoc:: /beginner/examples_autograd/polynomial_autograd.py
Under the hood, each primitive autograd operator is really two functions that operate on Tensors. The forward function computes output Tensors from input Tensors. The backward function receives the gradient of the output Tensors with respect to some scalar value, and computes the gradient of the input Tensors with respect to that same scalar value.
In PyTorch we can easily define our own autograd operator by defining a
subclass of torch.autograd.Function
and implementing the forward
and backward
functions. We can then use our new autograd operator by
constructing an instance and calling it like a function, passing
Tensors containing input data.
In this example we define our model as y=a+b P_3(c+dx) instead of y=a+bx+cx^2+dx^3, where P_3(x)=\frac{1}{2}\left(5x^3-3x\right) is the Legendre polynomial of degree three. We write our own custom autograd function for computing forward and backward of P_3, and use it to implement our model:
.. includenodoc:: /beginner/examples_autograd/polynomial_custom_function.py
Computational graphs and autograd are a very powerful paradigm for defining complex operators and automatically taking derivatives; however for large neural networks raw autograd can be a bit too low-level.
When building neural networks we frequently think of arranging the computation into layers, some of which have learnable parameters which will be optimized during learning.
In TensorFlow, packages like Keras, TensorFlow-Slim, and TFLearn provide higher-level abstractions over raw computational graphs that are useful for building neural networks.
In PyTorch, the nn
package serves this same purpose. The nn
package defines a set of Modules, which are roughly equivalent to
neural network layers. A Module receives input Tensors and computes
output Tensors, but may also hold internal state such as Tensors
containing learnable parameters. The nn
package also defines a set
of useful loss functions that are commonly used when training neural
networks.
In this example we use the nn
package to implement our polynomial model
network:
.. includenodoc:: /beginner/examples_nn/polynomial_nn.py
Up to this point we have updated the weights of our models by manually
mutating the Tensors holding learnable parameters with torch.no_grad()
.
This is not a huge burden for simple optimization algorithms like stochastic
gradient descent, but in practice we often train neural networks using more
sophisticated optimizers like AdaGrad
, RMSProp
, Adam
, and other.
The optim
package in PyTorch abstracts the idea of an optimization
algorithm and provides implementations of commonly used optimization
algorithms.
In this example we will use the nn
package to define our model as
before, but we will optimize the model using the RMSprop
algorithm provided
by the optim
package:
.. includenodoc:: /beginner/examples_nn/polynomial_optim.py
Sometimes you will want to specify models that are more complex than a
sequence of existing Modules; for these cases you can define your own
Modules by subclassing nn.Module
and defining a forward
which
receives input Tensors and produces output Tensors using other
modules or other autograd operations on Tensors.
In this example we implement our third order polynomial as a custom Module subclass:
.. includenodoc:: /beginner/examples_nn/polynomial_module.py
As an example of dynamic graphs and weight sharing, we implement a very strange model: a third-fifth order polynomial that on each forward pass chooses a random number between 3 and 5 and uses that many orders, reusing the same weights multiple times to compute the fourth and fifth order.
For this model we can use normal Python flow control to implement the loop, and we can implement weight sharing by simply reusing the same parameter multiple times when defining the forward pass.
We can easily implement this model as a Module subclass:
.. includenodoc:: /beginner/examples_nn/dynamic_net.py
You can browse the above examples here.
.. toctree:: :maxdepth: 2 :hidden: /beginner/examples_tensor/polynomial_numpy /beginner/examples_tensor/polynomial_tensor
.. galleryitem:: /beginner/examples_tensor/polynomial_numpy.py
.. galleryitem:: /beginner/examples_tensor/polynomial_tensor.py
.. toctree:: :maxdepth: 2 :hidden: /beginner/examples_autograd/polynomial_autograd /beginner/examples_autograd/polynomial_custom_function
.. galleryitem:: /beginner/examples_autograd/polynomial_autograd.py
.. galleryitem:: /beginner/examples_autograd/polynomial_custom_function.py
.. toctree:: :maxdepth: 2 :hidden: /beginner/examples_nn/polynomial_nn /beginner/examples_nn/polynomial_optim /beginner/examples_nn/polynomial_module /beginner/examples_nn/dynamic_net
.. galleryitem:: /beginner/examples_nn/polynomial_nn.py
.. galleryitem:: /beginner/examples_nn/polynomial_optim.py
.. galleryitem:: /beginner/examples_nn/polynomial_module.py
.. galleryitem:: /beginner/examples_nn/dynamic_net.py