{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# Introduction to PyTorch\n", "> In this first chapter, we introduce basic concepts of neural networks and deep learning using PyTorch library. This is the Summary of lecture \"Introduction to Deep Learning with PyTorch\", via datacamp.\n", "\n", "- toc: true \n", "- badges: true\n", "- comments: true\n", "- author: Chanseok Kang\n", "- categories: [Python, Datacamp, PyTorch, Deep_Learning]\n", "- image: images/graph_exercise.jpg" ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [], "source": [ "import torch\n", "import torch.nn as nn\n", "import numpy as np\n", "import pandas as pd\n", "import matplotlib.pyplot as plt" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Introduction to PyTorch" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Creating tensors in PyTorch\n", "Random tensors are very important in neural networks. Parameters of the neural networks typically are initialized with random weights (random tensors).\n", "\n", "Let us start practicing building tensors in PyTorch library. As you know, tensors are arrays with an arbitrary number of dimensions, corresponding to NumPy's ndarrays. You are going to create a random tensor of sizes 3 by 3 and set it to variable `your_first_tensor`. Then, you will need to print it. Finally, calculate its size in variable `tensor_size` and print its value." ] }, { "cell_type": "code", "execution_count": 2, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "tensor([[0.5816, 0.6053, 0.8546],\n", " [0.4350, 0.1575, 0.9543],\n", " [0.8456, 0.5518, 0.9431]])\n", "torch.Size([3, 3])\n" ] } ], "source": [ "# Create random Tensor of size 3x3\n", "your_first_tensor = torch.rand(3, 3)\n", "\n", "# Calcualtethe shape of the tensor\n", "tensor_size = your_first_tensor.shape\n", "\n", "# Print the values of the tensor and its shape\n", "print(your_first_tensor)\n", "print(tensor_size)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Matrix multiplication\n", "There are many important types of matrices which have their uses in neural networks. Some important matrices are matrices of ones (where each entry is set to 1) and the identity matrix (where the diagonal is set to 1 while all other values are 0). The identity matrix is very important in linear algebra: any matrix multiplied with identity matrix is simply the original matrix.\n", "\n", "Let us experiment with these two types of matrices. You are going to build a matrix of ones with shape 3 by 3 called `tensor_of_ones` and an identity matrix of the same shape, called `identity_tensor`. We are going to see what happens when we multiply these two matrices, and what happens if we do an element-wise multiplication of them." ] }, { "cell_type": "code", "execution_count": 3, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "tensor([[1., 1., 1.],\n", " [1., 1., 1.],\n", " [1., 1., 1.]])\n", "tensor([[1., 0., 0.],\n", " [0., 1., 0.],\n", " [0., 0., 1.]])\n" ] } ], "source": [ "# Create a matrix of ones with shape 3x3\n", "tensor_of_ones = torch.ones(3, 3)\n", "\n", "# Create a identity matrix with shape 3x3\n", "identity_tensor = torch.eye(3)\n", "\n", "# Do a matrix multiplication of tensor_of_ones with identity_tensor\n", "matrices_multiplied = torch.matmul(tensor_of_ones, identity_tensor)\n", "print(matrices_multiplied)\n", "\n", "# Do an element-wise multiplication of tensor_of_ones with identity_tensor\n", "element_multipliaction = tensor_of_ones * identity_tensor\n", "print(element_multipliaction)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "`matrices_multiplied` is same as `tensor_of_ones` (because identity matrix is the neutral element in matrix multiplication, the product of any matrix multiplied with it gives the original matrix), while element_multiplication is same as `identity_tensor`." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Forward propagation\n" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Forward pass\n", "Let's have something resembling more a neural network. The computational graph has been given below. You are going to initialize 3 large random tensors, and then do the operations as given in the computational graph. The final operation is the mean of the tensor, given by `torch.mean(your_tensor)`.\n", "![multilabel](image/mutilabel_dataset2.jpg)" ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "tensor(124.9414)\n" ] } ], "source": [ "# Intialize tensors x, y, and z\n", "x = torch.rand(1000, 1000)\n", "y = torch.rand(1000, 1000)\n", "z = torch.rand(1000, 1000)\n", "\n", "# Multiply x with y\n", "q = torch.matmul(x, y)\n", "\n", "# Multiply elementwise z with q\n", "f = z * q\n", "\n", "mean_f = torch.mean(f)\n", "print(mean_f)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Backpropagation by auto-differentiation\n", "\n" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Backpropagation using PyTorch\n", "Here, you are going to use automatic differentiation of PyTorch in order to compute the derivatives of `x`, `y` and `z` from the previous exercise." ] }, { "cell_type": "code", "execution_count": 5, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Gradient of x is: tensor(5.)\n", "Gradient of y is: tensor(5.)\n", "Gradient of z is: tensor(1.)\n" ] } ], "source": [ "# Initialize x, y and z to values 4, -3, and 5\n", "x = torch.tensor(4., requires_grad=True)\n", "y = torch.tensor(-3., requires_grad=True)\n", "z = torch.tensor(5., requires_grad=True)\n", "\n", "# Set q to sum of x and y, set f to product of q with z\n", "q = x + y\n", "f = q * z\n", "\n", "# Compute the derivative\n", "f.backward()\n", "\n", "# Print the gradients\n", "print('Gradient of x is: ' + str(x.grad))\n", "print('Gradient of y is: ' + str(y.grad))\n", "print('Gradient of z is: ' + str(z.grad))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Calculating gradients in PyTorch\n", "Remember the exercise in forward pass? Now that you know how to calculate derivatives, let's make a step forward and start calculating the gradients (derivatives of tensors) of the computational graph you built back then. We have already initialized for you three random tensors of shape (1000, 1000) called `x`, `y` and `z`. First, we multiply tensors `x` and `y`, then we do an elementwise multiplication of their product with tensor `z`, and then we compute its mean. In the end, we compute the derivatives.\n", "\n", "The main difference from the previous exercise is the scale of the tensors. While before, tensors `x`, `y` and `z` had just 1 number, now they each have 1 million numbers.\n", "![graph](image/graph_exercise.jpg)" ] }, { "cell_type": "code", "execution_count": 6, "metadata": {}, "outputs": [], "source": [ "x = torch.rand(1000, 1000, requires_grad=True)\n", "y = torch.rand(1000, 1000, requires_grad=True)\n", "z = torch.rand(1000, 1000, requires_grad=True)\n", "\n", "# Multiply tensors x and y\n", "q = torch.matmul(x, y)\n", "\n", "# Elementwise multiply tensors z with q\n", "f = z * q\n", "\n", "mean_f = torch.mean(f)\n", "\n", "# Calculate the gradients\n", "mean_f.backward()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Introduction to Neural Networks" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Your first neural network\n", "You are going to build a neural network in PyTorch, using the hard way. Your input will be images of size (28, 28), so images containing 784 pixels. Your network will contain an `input_layer`, a hidden layer with 200 units, and an output layer with 10 classes. The input layer has already been created for you. You are going to create the weights, and then do matrix multiplications, getting the results from the network." ] }, { "cell_type": "code", "execution_count": 7, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "tensor([18735.8730, 18511.5098, 20068.1992, 17833.7676, 17848.9141, 18220.7695,\n", " 19325.1348, 18821.3984, 19510.9082, 20627.5332])\n" ] } ], "source": [ "input_layer = torch.rand(784)\n", "\n", "# Initialize the weights of the neural network\n", "weight_1 = torch.rand(784, 200)\n", "weight_2 = torch.rand(200, 10)\n", "\n", "# Multiply input_layer with weight_1\n", "hidden_1 = torch.matmul(input_layer, weight_1)\n", "\n", "# Multiply hidden_1 with weight_2\n", "output_layer = torch.matmul(hidden_1, weight_2)\n", "print(output_layer)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Your first PyTorch neural network\n", "You are going to build the same neural network you built in the previous exercise, but now using the PyTorch way. As a reminder, you have 784 units in the input layer, 200 hidden units and 10 units for the output layer.\n", "\n" ] }, { "cell_type": "code", "execution_count": 8, "metadata": {}, "outputs": [], "source": [ "class Net(nn.Module):\n", " def __init__(self):\n", " super(Net, self).__init__()\n", " \n", " # Instantiate all 2 linear layers\n", " self.fc1 = nn.Linear(784, 200)\n", " self.fc2 = nn.Linear(200, 10)\n", " \n", " def forward(self, x):\n", " # Use the instantiated layers and return x\n", " x = self.fc1(x)\n", " x = self.fc2(x)\n", " return x" ] } ], "metadata": { "kernelspec": { "display_name": "Python 3", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.7.6" } }, "nbformat": 4, "nbformat_minor": 4 }