Skip to content

Instantly share code, notes, and snippets.

@nurujjamanpollob
Last active January 14, 2024 16:39
Show Gist options
  • Save nurujjamanpollob/9db7546a4a7900269af9fea0e8adde61 to your computer and use it in GitHub Desktop.
Save nurujjamanpollob/9db7546a4a7900269af9fea0e8adde61 to your computer and use it in GitHub Desktop.
C2W1_Assignment.ipynb
Display the source blob
Display the rendered blob
Raw
{
"cells": [
{
"cell_type": "markdown",
"metadata": {
"id": "view-in-github",
"colab_type": "text"
},
"source": [
"<a href=\"https://colab.research.google.com/gist/nurujjamanpollob/9db7546a4a7900269af9fea0e8adde61/c2w1_assignment.ipynb\" target=\"_parent\"><img src=\"https://colab.research.google.com/assets/colab-badge.svg\" alt=\"Open In Colab\"/></a>"
]
},
{
"cell_type": "markdown",
"metadata": {
"id": "AuW-xg_bTsaF"
},
"source": [
"# Week 1: Using CNN's with the Cats vs Dogs Dataset\n",
"\n",
"Welcome to the 1st assignment of the course! This week, you will be using the famous `Cats vs Dogs` dataset to train a model that can classify images of dogs from images of cats. For this, you will create your own Convolutional Neural Network in Tensorflow and leverage Keras' image preprocessing utilities.\n",
"\n",
"You will also create some helper functions to move the images around the filesystem so if you are not familiar with the `os` module be sure to take a look a the [docs](https://docs.python.org/3/library/os.html).\n",
"\n",
"Let's get started!"
],
"id": "AuW-xg_bTsaF"
},
{
"cell_type": "code",
"execution_count": 8,
"metadata": {
"id": "dn-6c02VmqiN",
"tags": [
"graded"
]
},
"outputs": [],
"source": [
"import os\n",
"import zipfile\n",
"import pathlib\n",
"import math\n",
"import random\n",
"import shutil\n",
"import tensorflow as tf\n",
"from tensorflow.keras.preprocessing.image import ImageDataGenerator\n",
"from shutil import copyfile\n",
"import matplotlib.pyplot as plt"
],
"id": "dn-6c02VmqiN"
},
{
"cell_type": "markdown",
"metadata": {
"id": "bLTQd84RUs1j"
},
"source": [
"Download the dataset from its original source by running the cell below. \n",
"\n",
"Note that the `zip` file that contains the images is unzipped under the `/tmp` directory."
],
"id": "bLTQd84RUs1j"
},
{
"cell_type": "code",
"execution_count": 9,
"metadata": {
"id": "3sd9dQWa23aj",
"lines_to_next_cell": 2,
"tags": [],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "56d1a780-fca2-4ea3-ef68-a93838a971b2"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"--2022-10-01 11:29:26-- https://download.microsoft.com/download/3/E/1/3E1C3F21-ECDB-4869-8368-6DEBA77B919F/kagglecatsanddogs_5340.zip\n",
"Resolving download.microsoft.com (download.microsoft.com)... 23.72.44.156, 2600:1413:a000:682::317f, 2600:1413:a000:6bb::317f\n",
"Connecting to download.microsoft.com (download.microsoft.com)|23.72.44.156|:443... connected.\n",
"HTTP request sent, awaiting response... 200 OK\n",
"Length: 824887076 (787M) [application/octet-stream]\n",
"Saving to: ‘/tmp/cats-and-dogs.zip’\n",
"\n",
"/tmp/cats-and-dogs. 100%[===================>] 786.67M 186MB/s in 4.5s \n",
"\n",
"2022-10-01 11:29:30 (176 MB/s) - ‘/tmp/cats-and-dogs.zip’ saved [824887076/824887076]\n",
"\n"
]
}
],
"source": [
"# If the URL doesn't work, visit https://www.microsoft.com/en-us/download/confirmation.aspx?id=54765\n",
"# And right click on the 'Download Manually' link to get a new URL to the dataset\n",
"\n",
"# Note: This is a very large dataset and will take some time to download\n",
"\n",
"!wget --no-check-certificate \\\n",
" \"https://download.microsoft.com/download/3/E/1/3E1C3F21-ECDB-4869-8368-6DEBA77B919F/kagglecatsanddogs_5340.zip\" \\\n",
" -O \"/tmp/cats-and-dogs.zip\"\n",
"\n",
"local_zip = '/tmp/cats-and-dogs.zip'\n",
"zip_ref = zipfile.ZipFile(local_zip, 'r')\n",
"zip_ref.extractall('/tmp')\n",
"zip_ref.close()"
],
"id": "3sd9dQWa23aj"
},
{
"cell_type": "markdown",
"metadata": {
"id": "e_HsUV9WVJHL"
},
"source": [
"Now the images are stored within the `/tmp/PetImages` directory. There is a subdirectory for each class, so one for dogs and one for cats."
],
"id": "e_HsUV9WVJHL"
},
{
"cell_type": "code",
"execution_count": 10,
"metadata": {
"id": "DM851ZmN28J3",
"tags": [
"graded"
],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "28adc815-120f-48eb-ceb6-2e098f86a8cc"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"There are 12500 images of dogs.\n",
"There are 12500 images of cats.\n"
]
}
],
"source": [
"source_path = '/tmp/PetImages'\n",
"\n",
"source_path_dogs = os.path.join(source_path, 'Dog')\n",
"source_path_cats = os.path.join(source_path, 'Cat')\n",
"\n",
"# Deletes all non-image files (there are two .db files bundled into the dataset)\n",
"!find /tmp/PetImages/ -type f ! -name \"*.jpg\" -exec rm {} +\n",
"\n",
"# os.listdir returns a list containing all files under the given path\n",
"print(f\"There are {len(os.listdir(source_path_dogs))} images of dogs.\")\n",
"print(f\"There are {len(os.listdir(source_path_cats))} images of cats.\")"
],
"id": "DM851ZmN28J3"
},
{
"cell_type": "markdown",
"metadata": {
"id": "G7dI86rmRGmC"
},
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"There are 12501 images of dogs.\n",
"There are 12501 images of cats.\n",
"```"
],
"id": "G7dI86rmRGmC"
},
{
"cell_type": "markdown",
"metadata": {
"id": "iFbMliudNIjW"
},
"source": [
"You will need a directory for cats-v-dogs, and subdirectories for training\n",
"and validation. These in turn will need subdirectories for 'cats' and 'dogs'. To accomplish this, complete the `create_train_val_dirs` below:"
],
"id": "iFbMliudNIjW"
},
{
"cell_type": "code",
"execution_count": 11,
"metadata": {
"cellView": "code",
"id": "F-QkLjxpmyK2",
"tags": [
"graded"
]
},
"outputs": [],
"source": [
"# Define root directory\n",
"root_dir = '/tmp/cats-v-dogs'\n",
"\n",
"# Empty directory to prevent FileExistsError is the function is run several times\n",
"if os.path.exists(root_dir):\n",
" shutil.rmtree(root_dir)\n",
"\n",
"# GRADED FUNCTION: create_train_val_dirs\n",
"def create_train_val_dirs(root_path):\n",
" \"\"\"\n",
" Creates directories for the train and test sets\n",
" \n",
" Args:\n",
" root_path (string) - the base directory path to create subdirectories from\n",
" \n",
" Returns:\n",
" None\n",
" \"\"\" \n",
" ### START CODE HERE\n",
"\n",
" # HINT:\n",
" # Use os.makedirs to create your directories with intermediate subdirectories\n",
" # Don't hardcode the paths. Use os.path.join to append the new directories to the root_path parameter\n",
"\n",
" # Define dir\n",
" training_dir = os.path.join(root_path, 'training')\n",
" validation_dir = os.path.join(root_path, 'validation')\n",
" training_cat_dir = os.path.join(training_dir, 'cats')\n",
" training_dogs_dir = os.path.join(training_dir, 'dogs')\n",
" validation_dog_dir = os.path.join(validation_dir, 'dogs')\n",
" validation_cat_dir = os.path.join(validation_dir, 'cats')\n",
"\n",
" # Create training directory\n",
" os.makedirs(training_dir)\n",
" os.makedirs(validation_dir)\n",
" os.makedirs(training_cat_dir)\n",
" os.makedirs(training_dogs_dir)\n",
" os.makedirs(validation_dog_dir)\n",
" os.makedirs(validation_cat_dir)\n",
"\n",
" pass\n",
"\n",
" ### END CODE HERE\n",
"\n",
" \n",
"try:\n",
" create_train_val_dirs(root_path=root_dir)\n",
"except FileExistsError:\n",
" print(\"You should not be seeing this since the upper directory is removed beforehand\")"
],
"id": "F-QkLjxpmyK2"
},
{
"cell_type": "code",
"execution_count": 6,
"metadata": {
"id": "5dhtL344OK00",
"tags": [
"graded"
],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "7a0d6d34-6873-4201-8965-b80debc5a9a5"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"/tmp/cats-v-dogs/validation\n",
"/tmp/cats-v-dogs/training\n",
"/tmp/cats-v-dogs/validation/cats\n",
"/tmp/cats-v-dogs/validation/dogs\n",
"/tmp/cats-v-dogs/training/cats\n",
"/tmp/cats-v-dogs/training/dogs\n"
]
}
],
"source": [
"# Test your create_train_val_dirs function\n",
"\n",
"for rootdir, dirs, files in os.walk(root_dir):\n",
" for subdir in dirs:\n",
" print(os.path.join(rootdir, subdir))"
],
"id": "5dhtL344OK00"
},
{
"cell_type": "markdown",
"metadata": {
"id": "D7A0RK3IQsvg"
},
"source": [
"**Expected Output (directory order might vary):**\n",
"\n",
"``` txt\n",
"/tmp/cats-v-dogs/training\n",
"/tmp/cats-v-dogs/validation\n",
"/tmp/cats-v-dogs/training/cats\n",
"/tmp/cats-v-dogs/training/dogs\n",
"/tmp/cats-v-dogs/validation/cats\n",
"/tmp/cats-v-dogs/validation/dogs\n",
"\n",
"```"
],
"id": "D7A0RK3IQsvg"
},
{
"cell_type": "markdown",
"metadata": {
"id": "R93T7HdE5txZ"
},
"source": [
"Code the `split_data` function which takes in the following arguments:\n",
"- SOURCE_DIR: directory containing the files\n",
"\n",
"- TRAINING_DIR: directory that a portion of the files will be copied to (will be used for training)\n",
"- VALIDATION_DIR: directory that a portion of the files will be copied to (will be used for validation)\n",
"- SPLIT_SIZE: determines the portion of images used for training.\n",
"\n",
"The files should be randomized, so that the training set is a random sample of the files, and the validation set is made up of the remaining files.\n",
"\n",
"For example, if `SOURCE_DIR` is `PetImages/Cat`, and `SPLIT_SIZE` is .9 then 90% of the images in `PetImages/Cat` will be copied to the `TRAINING_DIR` directory\n",
"and 10% of the images will be copied to the `VALIDATION_DIR` directory.\n",
"\n",
"All images should be checked before the copy, so if they have a zero file length, they will be omitted from the copying process. If this is the case then your function should print out a message such as `\"filename is zero length, so ignoring.\"`. **You should perform this check before the split so that only non-zero images are considered when doing the actual split.**\n",
"\n",
"\n",
"Hints:\n",
"\n",
"- `os.listdir(DIRECTORY)` returns a list with the contents of that directory.\n",
"\n",
"- `os.path.getsize(PATH)` returns the size of the file\n",
"\n",
"- `copyfile(source, destination)` copies a file from source to destination\n",
"\n",
"- `random.sample(list, len(list))` shuffles a list"
],
"id": "R93T7HdE5txZ"
},
{
"cell_type": "code",
"execution_count": 24,
"metadata": {
"cellView": "code",
"id": "zvSODo0f9LaU",
"tags": [
"graded"
]
},
"outputs": [],
"source": [
"# GRADED FUNCTION: split_data\n",
"def split_data(source_dir: str, training_dir: str, validation_dir: str, split_size: float):\n",
" \"\"\"\n",
" Splits the data into train and test sets\n",
"\n",
" Args:\n",
" source_dir (string): directory path containing the images\n",
" training_dir (string): directory path to be used for training\n",
" validation_dir (string): directory path to be used for validation\n",
" split_size (float): proportion of the dataset to be used for training\n",
"\n",
" Returns:\n",
" None\n",
" \"\"\"\n",
"\n",
" # Create file list\n",
" flist = []\n",
" for p in pathlib.Path(source_dir).iterdir():\n",
" if p.is_file():\n",
" flist.append(p)\n",
"\n",
" # randomize order\n",
" random.sample(flist, len(flist))\n",
"\n",
" # Calculate how many file to copy to training dir\n",
" training_file_count: int = math.floor((len(flist) / 100) * (split_size * 100))\n",
"\n",
" for f in flist:\n",
"\n",
" # Get file instance\n",
" file = os.path.join(f)\n",
"\n",
" # Get file size\n",
" file_size: int = os.path.getsize(file)\n",
"\n",
" # Get file name\n",
" file_name: str = os.path.basename(file)\n",
"\n",
" # If file size is empty\n",
" if file_size <= 0:\n",
"\n",
" # Show file skipped message\n",
" print(\"%s is zero length, so ignoring.\" % file_name)\n",
"\n",
" # Reduce training file count\n",
" training_file_count -= 1\n",
"\n",
" # File is for copy\n",
" else:\n",
"\n",
" # Copy to training directory\n",
" if training_file_count > 0:\n",
"\n",
" # copy\n",
" shutil.copy(os.path.join(file), os.path.join(training_dir))\n",
"\n",
" # Reduce training file count\n",
" training_file_count -= 1\n",
"\n",
" # Copy to validation directory\n",
" else:\n",
"\n",
" # Copy\n",
" shutil.copy(os.path.join(file), os.path.join(validation_dir))\n",
"\n",
" pass\n",
"\n",
" ### END CODE HERE\n"
],
"id": "zvSODo0f9LaU"
},
{
"cell_type": "code",
"execution_count": 25,
"metadata": {
"id": "FlIdoUeX9S-9",
"tags": [
"graded"
],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "714eb533-9238-4d31-d5b8-d681dbb1739a"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"666.jpg is zero length, so ignoring.\n",
"11702.jpg is zero length, so ignoring.\n",
"\n",
"\n",
"Original cat's directory has 12500 images\n",
"Original dog's directory has 12500 images\n",
"\n",
"There are 11249 images of cats for training\n",
"There are 11249 images of dogs for training\n",
"There are 1250 images of cats for validation\n",
"There are 1250 images of dogs for validation\n"
]
}
],
"source": [
"# Test your split_data function\n",
"\n",
"# Define paths\n",
"CAT_SOURCE_DIR = \"/tmp/PetImages/Cat/\"\n",
"DOG_SOURCE_DIR = \"/tmp/PetImages/Dog/\"\n",
"\n",
"TRAINING_DIR = \"/tmp/cats-v-dogs/training/\"\n",
"VALIDATION_DIR = \"/tmp/cats-v-dogs/validation/\"\n",
"\n",
"TRAINING_CATS_DIR = os.path.join(TRAINING_DIR, \"cats/\")\n",
"VALIDATION_CATS_DIR = os.path.join(VALIDATION_DIR, \"cats/\")\n",
"\n",
"TRAINING_DOGS_DIR = os.path.join(TRAINING_DIR, \"dogs/\")\n",
"VALIDATION_DOGS_DIR = os.path.join(VALIDATION_DIR, \"dogs/\")\n",
"\n",
"# Empty directories in case you run this cell multiple times\n",
"if len(os.listdir(TRAINING_CATS_DIR)) > 0:\n",
" for file in os.scandir(TRAINING_CATS_DIR):\n",
" os.remove(file.path)\n",
"if len(os.listdir(TRAINING_DOGS_DIR)) > 0:\n",
" for file in os.scandir(TRAINING_DOGS_DIR):\n",
" os.remove(file.path)\n",
"if len(os.listdir(VALIDATION_CATS_DIR)) > 0:\n",
" for file in os.scandir(VALIDATION_CATS_DIR):\n",
" os.remove(file.path)\n",
"if len(os.listdir(VALIDATION_DOGS_DIR)) > 0:\n",
" for file in os.scandir(VALIDATION_DOGS_DIR):\n",
" os.remove(file.path)\n",
"\n",
"# Define proportion of images used for training\n",
"split_size = .9\n",
"\n",
"# Run the function\n",
"# NOTE: Messages about zero length images should be printed out\n",
"split_data(CAT_SOURCE_DIR, TRAINING_CATS_DIR, VALIDATION_CATS_DIR, split_size)\n",
"split_data(DOG_SOURCE_DIR, TRAINING_DOGS_DIR, VALIDATION_DOGS_DIR, split_size)\n",
"\n",
"# Check that the number of images matches the expected output\n",
"\n",
"# Your function should perform copies rather than moving images so original directories should contain unchanged images\n",
"print(f\"\\n\\nOriginal cat's directory has {len(os.listdir(CAT_SOURCE_DIR))} images\")\n",
"print(f\"Original dog's directory has {len(os.listdir(DOG_SOURCE_DIR))} images\\n\")\n",
"\n",
"# Training and validation splits\n",
"print(f\"There are {len(os.listdir(TRAINING_CATS_DIR))} images of cats for training\")\n",
"print(f\"There are {len(os.listdir(TRAINING_DOGS_DIR))} images of dogs for training\")\n",
"print(f\"There are {len(os.listdir(VALIDATION_CATS_DIR))} images of cats for validation\")\n",
"print(f\"There are {len(os.listdir(VALIDATION_DOGS_DIR))} images of dogs for validation\")"
],
"id": "FlIdoUeX9S-9"
},
{
"cell_type": "markdown",
"metadata": {
"id": "hvskJNOFVSaz"
},
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"666.jpg is zero length, so ignoring.\n",
"11702.jpg is zero length, so ignoring.\n",
"\n",
"\n",
"Original cat's directory has 12500 images\n",
"Original dog's directory has 12500 images\n",
"\n",
"There are 11249 images of cats for training\n",
"There are 11249 images of dogs for training\n",
"There are 1250 images of cats for validation\n",
"There are 1250 images of dogs for validation\n",
"```"
],
"id": "hvskJNOFVSaz"
},
{
"cell_type": "markdown",
"metadata": {
"id": "Zil4QmOD_mXF"
},
"source": [
"Now that you have successfully organized the data in a way that can be easily fed to Keras' `ImageDataGenerator`, it is time for you to code the generators that will yield batches of images, both for training and validation. For this, complete the `train_val_generators` function below.\n",
"\n",
"Something important to note is that the images in this dataset come in a variety of resolutions. Luckily, the `flow_from_directory` method allows you to standarize this by defining a tuple called `target_size` that will be used to convert each image to this target resolution. **For this exercise, use a `target_size` of (150, 150)**.\n",
"\n",
"**Hint:** \n",
"\n",
"Don't use data augmentation by setting extra parameters when you instantiate the `ImageDataGenerator` class. This will make the training of your model to take longer to reach the necessary accuracy threshold to pass this assignment and this topic will be covered in the next week."
],
"id": "Zil4QmOD_mXF"
},
{
"cell_type": "code",
"execution_count": 29,
"metadata": {
"cellView": "code",
"id": "fQrZfVgz4j2g",
"tags": [
"graded"
]
},
"outputs": [],
"source": [
"# GRADED FUNCTION: train_val_generators\n",
"def train_val_generators(TRAINING_DIR, VALIDATION_DIR):\n",
" \"\"\"\n",
" Creates the training and validation data generators\n",
" \n",
" Args:\n",
" TRAINING_DIR (string): directory path containing the training images\n",
" VALIDATION_DIR (string): directory path containing the testing/validation images\n",
" \n",
" Returns:\n",
" train_generator, validation_generator - tuple containing the generators\n",
" \"\"\"\n",
" ### START CODE HERE\n",
"\n",
" # Instantiate the ImageDataGenerator class (don't forget to set the rescale argument)\n",
" train_datagen = ImageDataGenerator(rescale=1/255)\n",
"\n",
" # Pass in the appropiate arguments to the flow_from_directory method\n",
" train_generator = train_datagen.flow_from_directory(directory=TRAINING_DIR,\n",
" batch_size=128,\n",
" class_mode='binary',\n",
" target_size=(150, 150))\n",
"\n",
" # Instantiate the ImageDataGenerator class (don't forget to set the rescale argument)\n",
" validation_datagen = ImageDataGenerator(rescale=1/255)\n",
"\n",
" # Pass in the appropiate arguments to the flow_from_directory method\n",
" validation_generator = validation_datagen.flow_from_directory(directory=VALIDATION_DIR,\n",
" batch_size=128,\n",
" class_mode='binary',\n",
" target_size=(150, 150))\n",
" ### END CODE HERE\n",
" return train_generator, validation_generator\n"
],
"id": "fQrZfVgz4j2g"
},
{
"cell_type": "code",
"execution_count": 30,
"metadata": {
"id": "qM7FxrjGiobD",
"tags": [
"graded"
],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "695a2e09-4bd5-4e3b-b4ed-632fc1f606f9"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"Found 22498 images belonging to 2 classes.\n",
"Found 2500 images belonging to 2 classes.\n"
]
}
],
"source": [
"# Test your generators\n",
"train_generator, validation_generator = train_val_generators(TRAINING_DIR, VALIDATION_DIR)"
],
"id": "qM7FxrjGiobD"
},
{
"cell_type": "markdown",
"metadata": {
"id": "tiPNmSfZjHwJ"
},
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"Found 22498 images belonging to 2 classes.\n",
"Found 2500 images belonging to 2 classes.\n",
"```\n"
],
"id": "tiPNmSfZjHwJ"
},
{
"cell_type": "markdown",
"metadata": {
"id": "TI3oEmyQCZoO"
},
"source": [
"One last step before training is to define the architecture of the model that will be trained.\n",
"\n",
"Complete the `create_model` function below which should return a Keras' `Sequential` model.\n",
"\n",
"Aside from defining the architecture of the model, you should also compile it so make sure to use a `loss` function that is compatible with the `class_mode` you defined in the previous exercise, which should also be compatible with the output of your network. You can tell if they aren't compatible if you get an error during training.\n",
"\n",
"**Note that you should use at least 3 convolution layers to achieve the desired performance.**"
],
"id": "TI3oEmyQCZoO"
},
{
"cell_type": "code",
"execution_count": 36,
"metadata": {
"cellView": "code",
"id": "oDPK8tUB_O9e",
"lines_to_next_cell": 2,
"tags": [
"graded"
]
},
"outputs": [],
"source": [
"# GRADED FUNCTION: create_model\n",
"def create_model():\n",
" # DEFINE A KERAS MODEL TO CLASSIFY CATS V DOGS\n",
" # USE AT LEAST 3 CONVOLUTION LAYERS\n",
"\n",
" ### START CODE HERE\n",
"\n",
" model = tf.keras.models.Sequential([ \n",
" # Note the input shape is the desired size of the image 150x150 with 3 bytes color\n",
" # This is the first convolution\n",
" tf.keras.layers.Conv2D(8, (3,3), activation='relu', input_shape=(150, 150, 3)),\n",
" tf.keras.layers.MaxPooling2D(2, 2),\n",
" # The second convolution\n",
" tf.keras.layers.Conv2D(16, (3,3), activation='relu'),\n",
" tf.keras.layers.MaxPooling2D(2,2),\n",
" # The third convolution\n",
" tf.keras.layers.Conv2D(32, (3,3), activation='relu'),\n",
" tf.keras.layers.MaxPooling2D(2,2),\n",
" # Flatten the results to feed into a DNN\n",
" tf.keras.layers.Flatten(),\n",
" # 512 neuron hidden layer\n",
" tf.keras.layers.Dense(512, activation='relu'),\n",
" # Only 1 output neuron. It will contain a value from 0-1 where 0 for 1 class ('horses') and 1 for the other ('humans')\n",
" tf.keras.layers.Dense(1, activation='sigmoid')\n",
" ])\n",
"\n",
" from tensorflow.keras.optimizers import RMSprop\n",
"\n",
" model.compile(optimizer=RMSprop(learning_rate=0.001),\n",
" loss='binary_crossentropy',\n",
" metrics=['accuracy']) \n",
" \n",
" ### END CODE HERE\n",
"\n",
" return model\n"
],
"id": "oDPK8tUB_O9e"
},
{
"cell_type": "markdown",
"metadata": {
"id": "SMFNJZmTCZv6"
},
"source": [
"Now it is time to train your model!\n",
"\n",
"**Note:** You can ignore the `UserWarning: Possibly corrupt EXIF data.` warnings."
],
"id": "SMFNJZmTCZv6"
},
{
"cell_type": "code",
"execution_count": 37,
"metadata": {
"id": "5qE1G6JB4fMn",
"tags": [],
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "5de7bbb5-c781-4b13-a3de-43dd04352f6e"
},
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"Epoch 1/15\n",
" 18/176 [==>...........................] - ETA: 58s - loss: 1.2677 - accuracy: 0.5273"
]
},
{
"output_type": "stream",
"name": "stderr",
"text": [
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 32 bytes but only got 0. Skipping tag 270\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 5 bytes but only got 0. Skipping tag 271\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 8 bytes but only got 0. Skipping tag 272\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 8 bytes but only got 0. Skipping tag 282\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 8 bytes but only got 0. Skipping tag 283\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 20 bytes but only got 0. Skipping tag 306\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:770: UserWarning: Possibly corrupt EXIF data. Expecting to read 48 bytes but only got 0. Skipping tag 532\n",
" \" Skipping tag %s\" % (size, len(data), tag)\n",
"/usr/local/lib/python3.7/dist-packages/PIL/TiffImagePlugin.py:788: UserWarning: Corrupt EXIF data. Expecting to read 2 bytes but only got 0. \n",
" warnings.warn(str(msg))\n"
]
},
{
"output_type": "stream",
"name": "stdout",
"text": [
"176/176 [==============================] - 80s 431ms/step - loss: 0.6984 - accuracy: 0.6308 - val_loss: 0.5582 - val_accuracy: 0.7116\n",
"Epoch 2/15\n",
"176/176 [==============================] - 70s 397ms/step - loss: 0.5355 - accuracy: 0.7310 - val_loss: 0.5053 - val_accuracy: 0.7660\n",
"Epoch 3/15\n",
"176/176 [==============================] - 69s 394ms/step - loss: 0.4702 - accuracy: 0.7771 - val_loss: 0.5005 - val_accuracy: 0.7496\n",
"Epoch 4/15\n",
"176/176 [==============================] - 68s 387ms/step - loss: 0.4118 - accuracy: 0.8104 - val_loss: 0.6676 - val_accuracy: 0.6824\n",
"Epoch 5/15\n",
"176/176 [==============================] - 72s 411ms/step - loss: 0.3655 - accuracy: 0.8367 - val_loss: 0.4560 - val_accuracy: 0.7964\n",
"Epoch 6/15\n",
"176/176 [==============================] - 70s 396ms/step - loss: 0.3118 - accuracy: 0.8660 - val_loss: 0.5005 - val_accuracy: 0.7880\n",
"Epoch 7/15\n",
"176/176 [==============================] - 68s 386ms/step - loss: 0.2558 - accuracy: 0.8955 - val_loss: 0.7112 - val_accuracy: 0.7400\n",
"Epoch 8/15\n",
"176/176 [==============================] - 69s 390ms/step - loss: 0.1975 - accuracy: 0.9219 - val_loss: 0.5069 - val_accuracy: 0.8064\n",
"Epoch 9/15\n",
"176/176 [==============================] - 69s 392ms/step - loss: 0.1491 - accuracy: 0.9436 - val_loss: 0.5516 - val_accuracy: 0.8000\n",
"Epoch 10/15\n",
"176/176 [==============================] - 69s 391ms/step - loss: 0.1074 - accuracy: 0.9627 - val_loss: 0.6371 - val_accuracy: 0.8016\n",
"Epoch 11/15\n",
"176/176 [==============================] - 69s 393ms/step - loss: 0.0791 - accuracy: 0.9748 - val_loss: 1.4165 - val_accuracy: 0.7060\n",
"Epoch 12/15\n",
"176/176 [==============================] - 69s 391ms/step - loss: 0.0694 - accuracy: 0.9789 - val_loss: 0.8025 - val_accuracy: 0.8032\n",
"Epoch 13/15\n",
"176/176 [==============================] - 70s 396ms/step - loss: 0.0729 - accuracy: 0.9805 - val_loss: 0.8273 - val_accuracy: 0.8076\n",
"Epoch 14/15\n",
"176/176 [==============================] - 69s 392ms/step - loss: 0.0764 - accuracy: 0.9820 - val_loss: 0.7845 - val_accuracy: 0.7856\n",
"Epoch 15/15\n",
"176/176 [==============================] - 69s 392ms/step - loss: 0.0433 - accuracy: 0.9883 - val_loss: 1.0179 - val_accuracy: 0.8028\n"
]
}
],
"source": [
"# Get the untrained model\n",
"model = create_model()\n",
"\n",
"# Train the model\n",
"# Note that this may take some time.\n",
"history = model.fit(train_generator,\n",
" epochs=15,\n",
" verbose=1,\n",
" validation_data=validation_generator)"
],
"id": "5qE1G6JB4fMn"
},
{
"cell_type": "markdown",
"metadata": {
"id": "VGsaDMc-GMd4"
},
"source": [
"Once training has finished, you can run the following cell to check the training and validation accuracy achieved at the end of each epoch.\n",
"\n",
"**To pass this assignment, your model should achieve a training accuracy of at least 95% and a validation accuracy of at least 80%**. If your model didn't achieve these thresholds, try training again with a different model architecture and remember to use at least 3 convolutional layers."
],
"id": "VGsaDMc-GMd4"
},
{
"cell_type": "code",
"execution_count": 38,
"metadata": {
"id": "MWZrJN4-65RC",
"tags": [],
"colab": {
"base_uri": "https://localhost:8080/",
"height": 546
},
"outputId": "fcc68802-bc3f-43bf-ad5f-ae3c39ba7abf"
},
"outputs": [
{
"output_type": "display_data",
"data": {
"text/plain": [
"<Figure size 432x288 with 1 Axes>"
],
"image/png": "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\n"
},
"metadata": {
"needs_background": "light"
}
},
{
"output_type": "stream",
"name": "stdout",
"text": [
"\n"
]
},
{
"output_type": "display_data",
"data": {
"text/plain": [
"<Figure size 432x288 with 1 Axes>"
],
"image/png": "iVBORw0KGgoAAAANSUhEUgAAAa4AAAD4CAYAAAC0VQLEAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nO3dd3hUZdoG8PshxEAoQQVlASEqELBQQ5M60gQba0WxK34qu2svqN/aXdvavlUUWUVXFDsqCgaQYAGFhKpSLCBdcZFeQ57vj2eOGSAJKTPznjNz/65rriSTyTlPEpg773ueeV9RVRAREQVFFdcFEBERlQeDi4iIAoXBRUREgcLgIiKiQGFwERFRoFR1XUAyqFu3rmZmZroug4goUPLz839T1Xr73s/gioPMzEzk5eW5LoOIKFBE5Ofi7udUIRERBQqDi4iIAoXBRUREgcLgIiKiQGFwERFRoDC4iIgoUBhcREQUKAwuIkpa27cDo0YBe/a4roTKg8FFREnrzTeBoUOBnBzXlVB5MLiIKGl5C9pMneq2DiofBhcRJa38fHvL4AoWBhcRJaWCAmDuXOCgg4DZs4GNG11XRGXF4CKipLRokTVnXHghUFgIfP6564qorBhcRJSUvGnCYcOAtDROFwYJg4uIklJeHlCzJtCqFdClC4MrSBhcRJSU8vOBtm2BlBQgFLLrXevXu66KyoLBRURJx2vMaN/ePg6FAFXgs8/c1kVlw+AioqTjNWZ4wdWxI1C9OqcLg4LBRURJx2vM8IIrLQ3o2pXBFRQMLiJKOvn5QI0aQPPmRfeFQsCCBcC6de7qorJhcBFR0snLA9q1s8YMTyhkb6dNc1MTlR2Di4iSyr6NGZ7sbBuFcbrQ/xhcRJRU9m3M8KSmAt27M7iCgMFFREll38aMSKEQsHAhsHZtfGui8mFwEVFSKa4xw+Nd58rNjWtJVE4MLiJKKpErZuyrbVugdm1OF/odg4uIkkZBATBnjjViFKdqVaBHDwaX3zG4iChplNSYEalXL+D774FVq+JWFpUTg4uIkkZpjRke7zoXR13+xeAioqRRWmOGp3VroE4dBpefMbiIKGmU1pjhSUkBevZkcPkZg4uIkkJJK2YUJxQCli4Ffv459nVR+TG4iCgpLFoEbNtWckdhJF7n8jcGFxElhbI0ZniOOw449FC+ENmvGFxElBTK0pjhqVLF2uKnTrWdkclfGFxElBTK0pgRKRQCli+3a13kLwwuIkp45WnM8PA6l38xuIgo4XmNGeUJrpYtgcMPZ3D5EYOLiBKe15hRlo5Cjwivc/kVg4uIEl55GjMihULA6tW2diH5B4OLiBJeeRszPLzO5U8MLiJKaHv2lL8xw9OsGdCgAYPLbxhcRJTQKtKY4RGxUVduLq9z+QmDi4gSWl6eva1IcAEWXL/8AixcGL2aqHIYXESU0LzGjKysin09r3P5D4OLiBJaRRszPEceCTRuzODyEwYXESWsyjRmeCKvcxUWRq00qgQGFxElrMo0ZkQKhYD//hf45pvo1EWVw+AiooRVnq1MSsPrXP7C4CKihJWXV7nGDE/jxsBRRzG4/ILBRUQJq7KNGZF69QKmTbPrZuQWg4uIElI0GjMihULAhg3AvHnROR5VHIOLiBJStBozPLzO5R8MLiJKSNFqzPA0bGhrFzK43GNwEVFCquyKGcUJhYDPP7cdlckdBhcRJaS8PKBNm+g0ZnhCIWDTJmDOnOgdk8qPwUVECcdrzCjPjsdl0auXveV0oVsMLiJKONFuzPDUrw+0bMngco3BRUQJJ9qNGZG861y7d0f/2FQ2DC4iSjixaMzwhELA1q1F+3xR/DG4iCjh5OdHvzHDw+tc7jG4iCih7NljXX+xmCYEgLp1geOPZ3C5xOAiooTiNWZEu6MwUigEfPklsHNn7M6RCHbsiM1xSw0uEZkqIv33ue86ERlRytfkikh2+P2PRaROMY+5W0RuOsC5B4nIMREf3ysifUr7mrIQkV4iMr6yxyEif4plY4YnFAK2bwdmzozdOYJu6lTg6KOB+fOjf+wDjbheBzB4n/sGh+8/IFUdqKobKlIYgEEA/gguVf27qk6u4LGIKEnEsjHD07On7YzM6cLibdkCXHYZkJ4ONG0a/eMfKLjeBnCyiBwEACKSCaABgM9FZISI5InItyJyT3FfLCLLRKRu+P07RGSJiHwBICviMUNFZJaIzBORd0QkXUROAHAagEdFZK6IHC0io0XkrPDX9BaROSKyQEReFJG0iPPdIyKzw59rUdYfhIicF/6ab0Tk4fB9KeHzfhP+3PXh+/8mIt+JyHwRGVvWcxBR7MWyMcNz8MF2DgZX8W69Ffj5Z+Cllyy8oq3U4FLV9QBmAhgQvmswgDdVVQHcoarZAFoB6CkirUo6joi0D39tGwADAXSI+PS7qtpBVVsDWAjgclWdDuADADerahtV/THiWNUAjAZwrqoeD6AqgKsjjvebqrYDMAJAqdOREcdsAOBhACeGa+wgIoPC7zdU1ePC53op/CW3AWirqq0AXFXCMa8MB3veunXrylIGEVVSrBszIoVCwIwZsbuOE1Sffgo8+yxw7bVAt26xOUdZmjMipwsjpwnPEZHZAOYAOBYR03rF6A7gPVXdpqqbYKHkOU5EPheRBQCGhI9VmiwAS1V1SfjjlwH0iPj8u+G3+QAyD3AsTwcAuaq6TlULAIwJH/MnAEeJyP+JyEkANoUfPx/AGBG5AECxy22q6khVzVbV7Hr16pWxDCKqjFitmFGcUMiaM2bMiP25gmLLFuDyy20V/QceiN15yhJc7wPoLSLtAKSrar6IHAkbzfQOjzo+AlCtgjWMBvCX8Ijmnkocx+P1+eyBjcYqTFV/B9AaQC5sZDUq/KmTATwDoB2AWSJSqfMQUXR4jRmx7Cj0dO8OVKnC6cJIt9wS2ylCzwGDS1W3AJgK4EUUjbZqA9gKYKOIHI6iqcSSfAZgkIhUF5FaAE6N+FwtAGtEJBU24vJsDn9uX4sBZIqId8nvQgDTDvR9HMBM2HRnXRFJAXAegGnh63NVVPUdAHcCaCciVQAcoapTAdwKIANAzUqen4iiIB6NGZ6MDBvZMbjMp58CI0YA110HdO0a23OVdaTwOoD3EJ4yVNV5IjIHwCIAKwB8WdoXq+psEXkDwDwAvwKYFfHp/wXwNYB14bdeWI0F8IKI/A3AWRHH2iEilwJ4KzzSmQXguTJ+H57eIrIy4uOzYdetpgIQAB+p6vsi0hrAS+GwAoDhAFIAvCoiGeHHPl2JzkkiiqJ4NGZECoWAJ56w6clYjjD8bvNm6yJs1gy4//7Yn0+sz4JiKTs7W/O4sBlRTO3ZA9SuDVxxBfDUU/E558SJwIABQE4O0LdvfM7pR9dcAzz3nC0+HM3Rlojkh5sA98KVM4goIcSzMcPTrRtQtWpyTxd6U4TXXx/7KUIPg4uIEkI8VszYV82aQIcOyRtc3hRh8+bxmSL0MLiIKCHk59t1phZlXnYgOnr1AmbNsifxZHPLLcDy5cCLLwLVq8fvvAwuIkoI+flA27bxa8zwhEJ2fe3LUlvUEs+UKXZdK55ThB4GFxEFXjxXzNhX165AampyTRdu3mwvNI73FKGHL5wlosBbvDj+jRme9HSgU6fkCi5vivCLL+I7RejhiIuIAs97tYmL4AJsujA/H9i40c3548mbIrzhBuCEE9zUwOAiosBz1ZjhCYWAwkJ7HVMi86YIs7KA++5zVweDi4gCz1VjhqdLFyAtLfGnC2++GVixwtYidDFF6GFwEVGguWzM8FSrZuGVyME1eTLw/PM2Rdili9taGFxEFGguGzMihULA3LnA+vVu64iFTZuKpgjvvdd1NQwuIgo4FytmFCcUAlSBzz5zW0cs3HwzsHKl+ylCD4OLiAItL89tY4anY0d7Uk+06cLJk4GRI/0xRehhcBFRoLluzPCkpdmLkRMpuPw2RehhcBFRYPmhMSNSKAQsWACsW+e6kujwpghHj/bHFKGHwUVEgeWXxgxPKGRvp1V2T3YfmDTJpghvvBHo3Nl1NXtjcBFRYPmlMcOTnQ3UqBH86UJvirBFC39NEXq4ViERBZbrFTP2lZoKdO8em+DasMFGmAcfDDRrBohE/xyem24CVq0Cpk+316j5DYOLiAIrLw9o08Z9Y0akUAi49VZg7Vqgfv3yfa0qsHo1sHCh7ei8cGHRbe3aosc1aQL07w/06weceKKFWbTk5AAvvGAL6XbqFL3jRhODi4gCyWvMuOIK15XszbvOlZsLDB5c/GMKCoCffto7mBYtstumTUWPy8gAWrYEBgywt1lZFmyffAKMHWvXoKpUsYDp189uHTsCVSv4zL5pk/08W7QA7rmnYseIBwYXEQWS3xozPG3bArVr23ThqacCS5bsHVALFwLffw/s3l30NQ0aWDBddJG9bdnSwqN+/eKnBK+6yr5+5kwLsZwcW/T2nnss7Hr3thDr3x/IzCx77X6fIvSIqrquIeFlZ2drnrfvAhFFxX/+Y0/033wDHHus62r2duqpwIQJNir0VKkCHH10UTB5t6wsC5vKWr/ethzJybEwW7HC7m/WrCjEevUCatUq/utzcuwxt9wCPPxw5euJBhHJV9Xs/e5ncMUeg4so+q67zq7FbNrkr2tcgC37NGqUhYYXUE2b2ouU40HVRqReiOXm2ug0NdX20PKmFdu1s0DduBE4/nigZk1g9mz/jLYYXA4xuIiir3t32wPryy9dV+J/O3fa9J83rThnjt1/6KFA374W/hMnAjNm2DUyvygpuHiNi4gCZ88eGxlcfrnrSoIhLc2aRkIh4KGHgF9+sTUIc3LstnYtcNtt/gqt0jC4iChwvMaM7P3+FqeyOPxwYMgQu6lah+ORR7ququwYXEQUOH5bMSPIRKxpJEi45BMRBY7fVsyg+GJwEVHg5Of7b8UMih8GFxEFiteYwWnC5MXgIqJA8euKGRQ/DC4iChSvMYMdhcmLwUVEgcLGDGJwEQVA5Jp3yY6NGcTgIvK5994DDjkE+PBD15W4521lwutbyY3B5WeLFgFff+26CnJo6VLg0kttLbmLLrKPk9nixcDWrQyuZMfg8itV29Gtf/+iFTEpqezeDZx3nr0/ZYr9kzj7bFswNVlxxQwCGFz+JQK89ppt1NOvH/Ddd64roji74w4bcP/737Y9++jR9sR9/fWuK3OHjRkEMLj8rXFj+1M7NRXo0wf44QfXFVGcTJgAPPoocPXVwJln2n2DBtkOtSNG2N80ychrzKjo1vSUGBhcfte0qe0/sHu37ce9fLnriijGVq2y61mtWgGPP7735x58EOjWDbjyStsCPpmwMYM8DK4gOOYY2zRn40YLrzVrXFcUN3fdZdNCyfIt79kDXHABsH078MYb++9Em5oKjB1r02VnnWWNCsliyRI2ZpBhcAVF27Y2f7RmjU0b/vab64pi7pFHgHvvtU6yCy5Ijtcy3X+/bbP+7LMlX8dp2NCmChcuBK66ypo2koG3iTiDixhcQdKlCzB+vO361q8fsGGD64piZuRI4NZbgcGD7f1PP7WdWxNZbq4F9UUX2a00ffoAd98NvPoq8MIL8ajOvfx8oHp1NmYQIJosf645lJ2drXnen4vRMHEicNpptlhbTg5Qs2b0ju0Db7xhbeADBgDjxtmF+CFDgDfftCf3bt1cVxh969YBrVsDtWvbyKIsv9LCQvsZTZsGTJ8OtGsX+zpd6t7dvucvv3RdCcWLiOSr6n6rUnLEFUQnnWTP7jNnWoBt3+66oqiZOBG48EILp7fesms6IsBzzwGZmcD55wPr17uuMroKC4GLL7bv6803y/53SJUqNuKqW9de35XAA3Dk53MrEyrC4AqqP/8ZeOUVG4KcdRawa5friirtiy+AM84AjjvOljdKTy/6XO3a1pSwdi1w2WWJdV3nn/+0y5dPPGGdhOVRr56F3fLltsJGIv1cAGDZMhttZ2fbv4cDTaFSklBV3mJ8a9++vcbMCy+oAqpnnqm6e3fszhNjc+aoZmSoNm+u+ssvJT/u8cft23366fjVFkszZqhWrWq/vsLCih/H+7k89lj0anNp/XrVm25SPegg1WrVVIcPV92wwXVVFG8A8rSY51TnT+rJcItpcKmqPvmk/SovuEB1z57YnisGFi9WPeww1SOOUP3559IfW1ioesop9oQ2e3Z86ouV9etVmzRRzcxU/f33yh2rsFD1jDNUU1JUv/giKuU5sWOHhe/BB6uKqF5yieqKFa6rIldKCi5OFSaCa68FHnjALnhcfXWg5otWrAD69rWSJ02yxUJKIwK89JJNkZ17LrB5c3zqjDYNL0W5apVdrqxTp3LHEwFefNGuA557LvDrr1EpM24KC4HXX7eOwZtuAjp2tBcbv/QS0KiR6+rIbxhcieL22+02ciRwww2BCK9164q6+idOBLKyyvZ1desCY8YAP/4IDBsW2xpjZcQI4N13gX/8w56koyEjA3j7bXuJ35AhwXndW24u0KmTNd7UqWONshMnWpclUbGKG4bxFrCpQk9hoeq119q04R13xOecFbRxo2r79nb9Ytq0ih3j7rvtWx09Orq1xdqcOappaaoDB8ZmZte77HnXXdE/djR9+61N+wI2Tfzyy4Gc6aYYAq9xJUFwqVp4DR1qv9oHH4zfecth2zbVHj2sKeGjjyp+nIIC1Z49VdPTVRctilp5MbV5szWgNGigum5dbM5RWKh60UV2jeiTT2JzjspYvdr+iVapolq7tupDD9m/CaJ9MbiSJbhU7Rl9yBD79T71VHzPfQC7dtlf2SKqr71W+eOtXKl66KGqrVurbt9e+ePF2oUX2hN2bm5sz7Nli+qxx6rWreuf5obNm1X//nf7QyM11SYHYhXelBgYXMkUXKrWGv/nP9uveNSo+J+/GHv2qJ5/vpU0YkT0jjt+vB1z2LDoHTMWRo+2Ou++Oz7nW7RItWZN1RNOsD8YXNm9237fhx9u3/8556j+8IO7eig4GFzJFlyq1lt80kk2vBkzxk0NYYWFqtdcozGbwbzhBjv2u+9G/9jR8N13NtLo1csGxPEydqz9XG64IX7n9BQWqo4bp5qVZTV066b61Vfxr4OCi8EVwOC6/XZ7HcuECZX4i3nbNnu2TElx+qx+5532r+3mmyv3QtuS7Nypmp2tWqeO6rJl0T9+ZWzbpnr88ar16qmuWhX/8w8bFv9Q/+or1e7d7bxZWRZgsfi9U2JjcAUwuG680S5eA3YdZ+hQ1cmTK7BAxqZNqp0724WFCRMqVEtl/POf9j1ccUVsn7x++EG1Vi3VLl3cTo3t66qr7Pt38KNXVRt4d+hg/5ZiOUX344+q//qXav/+9v0edphNEfrpd0HBwuAKYHCpWsPBuHGq552nWqNG0RPCNddYG3mZ24d//121bVvrP586tcL1lNe//201n312fKbIXn/dzjd8eOzPVRZvvql/jDRdWrrUVqNo0yZ6HXzbt1vX4nXXWaekvXhQtWlTa8XftCk656HkxeAKaHBF2rpV9a23VM86S7V6dfvtNWhg3VnTp5dhNLNuneoxx1gCvv56zP8Ufvtt66Dr18+m8uLliivssl5OTvzOWZwff7RRTufO/hh1eE0sQ4dW/BhLl6o++6x1hqan2/HS0uxS6lNPqS5ZErVyiRhcLm+xaM7YvNnayU8/3dbtA1QbN7aFSWfNKiXEVq+2Pmkv9e69V3XNmqjXl5NjdZ1wgrVmx9PWraotW1oX29q18T23Z+dOm56rU8ee7P3ittvsV//KK2V7/I4dqpMmWXNHixb6x6jqyCPt2tlHH9nPmygWGFwJFlyRNmywVQcGDrQX9QKqRx1l02Vz5xYTYgUFqh98YEMhwK59DR6s+vnnUbkINX26/TXeqpUtJOvCggU2K9q3r5vVGLwux3feif+5S7N7d9GLtr/5pvjHLFtm16ZOO61oevqgg+xn+cQT1mbPRguKBwZXAgdXpP/+11621bevNRJ6XV1//7stsbOfxYvtIkVGhj24dWvVkSMrPEyaN89GGU2buhvteJ5/3r6lf/wjvuf1++vKVq+20WhWll2H2rlTdcoUG60fc4z+Mapq0kT16qvtb5x4j5qJVEsOLrHPUSxlZ2drXl5e3M+7bh3wzjtFW96r2iaN554LtGwJVKtWdEsr3I5qk8ej2tjRqLZkHqrVTkO1C89G2rArULVFU4gc+Hw//GDbq6ek2KaQmZmx/g5Lp2rf67vvAp9/DnTpEvtzrlwJtGkDHHEEMGOG/Wz9KDcX6N0baNbMVqjfssV2m+7RAxg4EBgwwFZqL8vvnShWRCRfVbP3u5/BFXuugivS2rW2cvgbb1iolEcV7EG1NEW1GimoVk2Kwi5t7/CbNw/YudNComXL2Hwf5bVxowVJYSEwdy5w8MGxOc9vvwGffAI8/jiweLFtM9+8eWzOFS1PPgk884wF2IABwIknArVqua6KqAiDyyE/BFekX36x/Zp27Nj7tnPnPvf9ugk7vsjDzlnzsGNLAXbUPgw7WrTBjiYtsBNp+339QQfZk2H2fv/M3Jo5E+jaFTjtNAvvaIwiCguBvDxgwgTg44+BWbNshFevnm1ZcuaZlT8HUbJjcDnkt+Aqt927gffesz/PP/vMhlfnnQf85S9Au3auqyuTxx4Dbr7ZvoVrrqnYMbxR1YQJ9va33ywEO3WyEcuAAUD79kAV7nJHFBUMLocCH1yRFiywZ////AfYtg3o3Nl2czz7bJs79KnCQuCUU4BPPwW+/rpsmxSWNqrq39+Cql8/29iSiKKPweVQQgWXZ8MG4OWXgWefBZYssWfzoUOByy4Djj7adXXF+vVXu95Vu7YFUs2a+z+mpFFVx45FTQscVRHFB4PLoYQMLk9hITBlCvCvfwHjx9vHPXoAl1wCnHWW7672f/op0KcPcPHFwEsv7T2qmjDBroep2ijqpJM4qiJyicHlUEIHV6SVK20KcfRoG4Wlp1t4XXIJ0LOnb4Yp//u/wP33WyjNmrX3qGrAABtZcVRF5B6Dy6GkCS6PKvDVVxZgY8cCmzYBTZrYMOeii5xPJRYU2DWq+fM5qiLyMwaXQ0kXXJG2bwfGjbMQmzTJQq17dxuFnX2206lEVb7AlsjPSgouToZQbFWvbq3zn3wCLF8OPPigvZDs8suB+vVtBDZ1ql1sijOGFlEwMbgofho1AoYPBxYtAqZPBy64AHj/fVuy4aijgLvuAn780XWVRORzDC6KPxFbOPD5520tqtdeA7KygPvuA5o2ta7EF18ENm92XSkR+RCDi9wqy1TipEnArl2uKyUin2BwkX9ETiXOmAFceCHwwQfW8levHnDOOcArr9iy90SUtNhVGAdJ3VVYWdu322jso4/sBc5r19pUY+fOtobTKacAxx/PTguiBMR2eIcYXFFSWAjMmWMBNn68LXkB2OZXXoiFQjb9SESBx+ByiMEVI2vW2Oq348fbdbCtWy20+vSxEDv5ZKBhQ9dVElEFMbgcYnDFwc6dwLRpFmIffggsW2b3t21bNBrLzuY6TkQBwuByiMEVZ6rAwoVFU4pffmnTjIcdZqOwU04B+vb13QLARLQ3BpdDDC7H1q8HJk60EJswwbZkSU21PesHDwYGDQIyMlxXSUT7YHA5xODykYICW7Xjww+Bt9+2KcW0NFsSfvBgG42lp7uukojAtQqJTNWqtjLHo48CP/1kq9hfdZW9Pfdcm04cMsSCjS96JvIlBhclLxGgUyfgySeBFStsl8khQ2xa8bTTgMMPB664Apg8Gdizx3W1RBTG4CICgJQUew2Yt37ixx8Dp54KvPmmNXI0bAj89a9FjR5E5AyDi2hfqam2u+Qrr9i6ie+8Y3uIjRoFdOsGHHkkcMstwOzZ1sFIRHHF4CIqTfXqwBlnAG+9Bfz6K/Dqq7bE1BNPAO3bAy1a2HYsCxe6rpQoaTC4iMqqVi27BjZ+vI3EXnjBFga+7z7gmGOANm2Ahx4qevEzEcUEg4uoIg45xBo3pkwBVq0Cnn7a2uiHD7epxB49gJEjgd9/d10pUcJhcBFV1p/+ZI0b06cDS5cCDzxgW6/8z//YnmJnnAG8+64tS0VElcbgIoqmzEzg9tuB774D8vOBYcNsb7Ezz7QQu/JK4LPP2JlIVAkMLqJYEAHatQMef9xeI/bJJ9Ze/9prQM+eNp3oBRwRlQuDiyjWqla1XZy99voxY4BjjwUeecTeegG3erXrSokCgcFFFE81agDnn28vcF692po6UlOBG2+0DTG9gNu82XWlRL7F4CJy5bDDrKnj66+BxYuBO+8EfvwRuPhiW27q/POBjz4Cdu92XSmRrzC4iPygeXPgnnuAH36w7sRLLwVycmy1+gYNLODy8rhSBxEYXET+IgJ06QI884xNJX7wAXDiifZi5w4dbNWOxx4D1qxxXSmRMwwuIr866CDrRHzjDVv49/nngdq1gZtvtuthJ59sS1Ht2OG6UqK4YnARBUGdOvYasOnTgUWLgFtvBebPB845x6YShw0DZs3iVCIlBQYXUdBkZdnqHMuW2XWwAQOAF18EOnYEjjvONsnkVCIlMAYXUVClpNheYWPG2FTiyJHAwQfbliuNGgEDB9p+YpxKpATD4CJKBBkZwNChwBdfWGv98OHAggXAuefaWorXXAPMnMmpREoIDC6iRNO8OXD//TaVOGmSNXGMHg106mTbrzz8sK1oTxRQDC6iRJWSAvTpY5tfrl1rOzjXrQvcdhvQuLFdG3vtNWDjRteVEpULg4soGdSuDVx+OfD558CSJbbA77ff2saY9eoB/fsDzz4LrFzpulKiAxLlnHfMZWdna15enusyiPZWWAh89RUwbhzw/vsWaACQnQ2cfjowaJAtAizitk5KWiKSr6rZ+93P4Io9BhcFwqJFRSH21Vd231FHWYCdfjrQtatNPxLFCYPLIQYXBc6aNcCHH1qQTZkC7Npl18dOOcVCrF8/ID3ddZWU4BhcDjG4KNA2b7aNMMeNs9XqN2wAqle315ANGmRhVq+e6yopATG4HGJwUcLYvRv47DObThw3znZ3rlLFphFPP91uTZu6rpISBIPLIQYXJSRVYO7couti8+bZ/ccea0YgZMkAAAnpSURBVK8d69sX6NYNqFbNbZ0UWAwuhxhclBSWLbMAe/99a7svKLDQ6t7dXk/Wty/QurWN0IjKgMHlEIOLks6WLcC0acDkybZ6x7ff2v116wK9e1uI9ekDNGnitk7yNQaXQwwuSnpr1hSF2OTJRavXN2tWFGKhkG3fQhTG4HKIwUUUQRX47ruiEMvNBbZutSnEjh2LphU7d7bNNClpMbgcYnARlWLXLuDrry3IJk2yVewLC4EaNYCePYtGZFzFI+kwuBxicBGVw4YNNgrzRmTeUlT169sK961aWZNHq1bA0Uez2SOBMbgcYnARVcLy5RZgU6YAc+bYfmOFhfa59HTg+OOLgqx1a/s4I8NtzRQVDC6HGFxEUbR9u10jmzfPbvPn29vffy96TGZmUZBxdBZYJQVXVRfFEBFVWPXqQPv2dvOo2uaYkUE2fz4wfjxHZwmII6444IiLyJHI0ZkXaPuOzho1srUW69TZ+5aRsf99kbdateI/glO1F3bv2mW33buL3q/MfQUFtnN2x442WvVJEwxHXESUfEobnXlBtnChBdmGDcD339vbDRvsRdSlEdk/3CI/rl5973DYNzzK+7F3X6zVq2cB1qmTve3QATjkkNiftxwYXESUXERslNWoETBwYMmPKygANm4sCrING/b/eN/bTz8Vvb99u70Ozbulpu79ceR9NWqU/LjIj1NTgbS04o91oHOUdj9gq5vMnGkvTZg5E/j4Ywt5wF4o3rFjUaC1bu10DUpOFcYBpwqJKHA2bQLy8izEvEBbvdo+l5pq4eWNyjp2tKnGKE+dsqvQIQYXESWEVav2HpXNmlU0pZqRYdOKkSOz+vUrdToGl0MMLiJKSHv2AIsW7R1m8+fb/QBwxBHAhAm26kkFsDmDiIiiKyXFQunYY4FLL7X7tm2zF4p7Yda4cdRPy+AiIqLoSU+3HbG7do3ZKfgyciIiCpSoBJeIHCoic8O3tSKyKuLjUvclEJFsEXm6DOeYHqVae4nI+Ggci4iI4i8qU4Wq+l8AbQBARO4GsEVVH/M+LyJVVbWghK/NA3DAzgVVPSEatRIRUbDFbKpQREaLyHMi8jWAR0Sko4jMEJE5IjJdRLLCj/tjBCQid4vIiyKSKyI/icjfIo63JeLxuSLytogsEpExIrY+iYgMDN+XLyJPl2dkJSLnicgCEflGRB4O35cS/j6+CX/u+vD9fxOR70RkvoiMjdoPjYiIDijWzRmNAJygqntEpDaA7qpaICJ9ADwI4MxivqYFgBCAWgAWi8gIVd13nZO2AI4FsBrAlwC6ikgegOcB9FDVpSLyelmLFJEGAB4G0B7A7wByRGQQgBUAGqrqceHHefuK3wbgSFXdGXHfvse8EsCVANA4Bl01RETJKtbNGW+parihHxkA3hKRbwA8AQue4nykqjtV9TcAvwI4vJjHzFTVlapaCGAugExY4P2kqkvDjylzcAHoACBXVdeFpzTHAOgB4CcAR4nI/4nISQA2hR8/H8AYEbkAQElToCNVNVtVs+vVq1eOUoiIqDSxDq6tEe/fB2BqePRyKoCSFrraGfH+HhQ/KizLYypNVX8H0BpALoCrAIwKf+pkAM8AaAdglojwZQVERHESz3b4DACrwu9fEoPjL4aNjjLDH59bjq+dCaCniNQVkRQA5wGYJiJ1AVRR1XcA3AmgnYhUAXCEqk4FcCvs+6oZpe+BiIgOIJ4jhUcAvCwidwL4KNoHV9XtInINgIkishXArFIe3ltEVkZ8fDbsutVUAAKbrnxfRFoDeCkcVgAwHEAKgFdFJCP82KdVdUO0vx8iIipeQq1VKCI1VXVLuMvwGQDfq+oTruviWoVEROVX0lqFibZyxlARmQvgW9gU3vOO6yEioihLqKaC8OjK+QiLiIhiJ9FGXERElOAYXEREFCgJ1ZzhVyKyDsDPFfzyugB+i2I5sRakellr7ASp3iDVCgSr3srW2kRV91vBgcHlcyKSV1xXjV8FqV7WGjtBqjdItQLBqjdWtXKqkIiIAoXBRUREgcLg8r+RrgsopyDVy1pjJ0j1BqlWIFj1xqRWXuMiIqJA4YiLiIgChcFFRESBwuDyMRE5SUQWi8gPInKb63pKIiJHiMhUEflORL4VkWtd13QgIpIiInNEZLzrWg5EROqIyNsiskhEFopIF9c1lURErg//G/hGRF4XkZL23XNCRF4UkV/DG9p69x0iIpNE5Pvw24Nd1hiphHofDf9bmC8i75W0C3u8FVdrxOduFBENbxVVaQwunwrvC/YMgAEAjgFwnogc47aqEhUAuFFVjwHQGcAwH9fquRbAQtdFlNFTACaqagvYxqa+rFtEGgL4G4Ds8IaxKQAGu61qP6MBnLTPfbcBmKKqzQBMCX/sF6Oxf72TABynqq0ALIFtt+QHo7F/rRCRIwD0A7A8WidicPlXRwA/qOpPqroLwFgApzuuqViqukZVZ4ff3wx7Ym3otqqSiUgj2C7Wow70WNfC+771APBvAFDVXT7f/60qgOrhXcHTAax2XM9eVPUzAOv3uft0AC+H338ZwKC4FlWK4upV1RxVLQh/+BWARnEvrBgl/GwBW/j8FgBR6wRkcPlXQwArIj5eCR+HgSe8A3VbAF+7raRUT8L+IxW6LqQMjgSwDrah6RwRGSUiNVwXVRxVXQXgMdhf1msAbFTVHLdVlcnhqrom/P5aAIe7LKacLgMwwXURJRGR0wGsUtV50Twug4uiRkRqAngHwHWqusl1PcURkVMA/Kqq+a5rKaOqANoBGKGqbQFshb+msv4QvjZ0OixsGwCoISIXuK2qfNReHxSI1wiJyB2wafoxrmspjoikA7gdwN+jfWwGl3+tAnBExMeNwvf5koikwkJrjKq+67qeUnQFcJqILINNv54oIq+6LalUKwGsVFVvBPs2LMj8qA+Apaq6TlV3A3gXwAmOayqLX0TkTwAQfvur43oOSEQuAXAKgCHq3xfjHg37I2Ze+P9bIwCzRaR+ZQ/M4PKvWQCaiciRInIQ7CL3B45rKpaICOwazEJVfdx1PaVR1eGq2khVM2E/009V1bejAlVdC2CFiGSF7+oN4DuHJZVmOYDOIpIe/jfRGz5tJNnHBwAuDr9/MYD3HdZyQCJyEmyq+zRV3ea6npKo6gJVPUxVM8P/31YCaBf+N10pDC6fCl98/QuAT2D/+d9U1W/dVlWirgAuhI1e5oZvA10XlUD+CmCMiMwH0AbAg47rKVZ4VPg2gNkAFsCeX3y1PJGIvA5gBoAsEVkpIpcDeAhAXxH5HjZqfMhljZFKqPdfAGoBmBT+v/ac0yLDSqg1Nufy7yiTiIhofxxxERFRoDC4iIgoUBhcREQUKAwuIiIKFAYXEREFCoOLiIgChcFFRESB8v8vHuhY7E2CBgAAAABJRU5ErkJggg==\n"
},
"metadata": {
"needs_background": "light"
}
}
],
"source": [
"#-----------------------------------------------------------\n",
"# Retrieve a list of list results on training and test data\n",
"# sets for each training epoch\n",
"#-----------------------------------------------------------\n",
"acc=history.history['accuracy']\n",
"val_acc=history.history['val_accuracy']\n",
"loss=history.history['loss']\n",
"val_loss=history.history['val_loss']\n",
"\n",
"epochs=range(len(acc)) # Get number of epochs\n",
"\n",
"#------------------------------------------------\n",
"# Plot training and validation accuracy per epoch\n",
"#------------------------------------------------\n",
"plt.plot(epochs, acc, 'r', \"Training Accuracy\")\n",
"plt.plot(epochs, val_acc, 'b', \"Validation Accuracy\")\n",
"plt.title('Training and validation accuracy')\n",
"plt.show()\n",
"print(\"\")\n",
"\n",
"#------------------------------------------------\n",
"# Plot training and validation loss per epoch\n",
"#------------------------------------------------\n",
"plt.plot(epochs, loss, 'r', \"Training Loss\")\n",
"plt.plot(epochs, val_loss, 'b', \"Validation Loss\")\n",
"plt.show()"
],
"id": "MWZrJN4-65RC"
},
{
"cell_type": "markdown",
"metadata": {
"id": "NYIaqsN2pav6"
},
"source": [
"You will probably encounter that the model is overfitting, which means that it is doing a great job at classifying the images in the training set but struggles with new data. This is perfectly fine and you will learn how to mitigate this issue in the upcoming week.\n",
"\n",
"Before downloading this notebook and closing the assignment, be sure to also download the `history.pkl` file which contains the information of the training history of your model. You can download this file by running the cell below:"
],
"id": "NYIaqsN2pav6"
},
{
"cell_type": "code",
"execution_count": 39,
"metadata": {
"id": "yWcrc9nZTsHj",
"tags": [],
"colab": {
"base_uri": "https://localhost:8080/",
"height": 17
},
"outputId": "b6ae9759-ed8d-436e-c3b0-80e6863a9793"
},
"outputs": [
{
"output_type": "display_data",
"data": {
"text/plain": [
"<IPython.core.display.Javascript object>"
],
"application/javascript": [
"\n",
" async function download(id, filename, size) {\n",
" if (!google.colab.kernel.accessAllowed) {\n",
" return;\n",
" }\n",
" const div = document.createElement('div');\n",
" const label = document.createElement('label');\n",
" label.textContent = `Downloading \"${filename}\": `;\n",
" div.appendChild(label);\n",
" const progress = document.createElement('progress');\n",
" progress.max = size;\n",
" div.appendChild(progress);\n",
" document.body.appendChild(div);\n",
"\n",
" const buffers = [];\n",
" let downloaded = 0;\n",
"\n",
" const channel = await google.colab.kernel.comms.open(id);\n",
" // Send a message to notify the kernel that we're ready.\n",
" channel.send({})\n",
"\n",
" for await (const message of channel.messages) {\n",
" // Send a message to notify the kernel that we're ready.\n",
" channel.send({})\n",
" if (message.buffers) {\n",
" for (const buffer of message.buffers) {\n",
" buffers.push(buffer);\n",
" downloaded += buffer.byteLength;\n",
" progress.value = downloaded;\n",
" }\n",
" }\n",
" }\n",
" const blob = new Blob(buffers, {type: 'application/binary'});\n",
" const a = document.createElement('a');\n",
" a.href = window.URL.createObjectURL(blob);\n",
" a.download = filename;\n",
" div.appendChild(a);\n",
" a.click();\n",
" div.remove();\n",
" }\n",
" "
]
},
"metadata": {}
},
{
"output_type": "display_data",
"data": {
"text/plain": [
"<IPython.core.display.Javascript object>"
],
"application/javascript": [
"download(\"download_e54ad047-ae34-4ccd-8510-d36962824064\", \"history.pkl\", 628)"
]
},
"metadata": {}
}
],
"source": [
"def download_history():\n",
" import pickle\n",
" from google.colab import files\n",
"\n",
" with open('history.pkl', 'wb') as f:\n",
" pickle.dump(history.history, f)\n",
"\n",
" files.download('history.pkl')\n",
"\n",
"download_history()"
],
"id": "yWcrc9nZTsHj"
},
{
"cell_type": "markdown",
"metadata": {
"id": "sjqp2UpkS_Cn"
},
"source": [
"You will also need to submit this notebook for grading. To download it, click on the `File` tab in the upper left corner of the screen then click on `Download` -> `Download .ipynb`. You can name it anything you want as long as it is a valid `.ipynb` (jupyter notebook) file."
],
"id": "sjqp2UpkS_Cn"
},
{
"cell_type": "markdown",
"metadata": {
"id": "joAaZSWWpbOI"
},
"source": [
"**Congratulations on finishing this week's assignment!**\n",
"\n",
"You have successfully implemented a convolutional neural network that classifies images of cats and dogs, along with the helper functions needed to pre-process the images!\n",
"\n",
"**Keep it up!**"
],
"id": "joAaZSWWpbOI"
}
],
"metadata": {
"accelerator": "GPU",
"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.4"
},
"colab": {
"provenance": [],
"include_colab_link": true
}
},
"nbformat": 4,
"nbformat_minor": 5
}
@nurujjamanpollob
Copy link
Author

C2W1_Assignment solution for Cats vs Dog

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment