Skip to content

Instantly share code, notes, and snippets.

@johnleung8888
Last active April 21, 2024 11:26
Show Gist options
  • Save johnleung8888/ecc6003207268c94d204d2fc7fc1f004 to your computer and use it in GitHub Desktop.
Save johnleung8888/ecc6003207268c94d204d2fc7fc1f004 to your computer and use it in GitHub Desktop.
submissionC2W1.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/johnleung8888/ecc6003207268c94d204d2fc7fc1f004/submissionc2w1.ipynb\" target=\"_parent\"><img src=\"https://colab.research.google.com/assets/colab-badge.svg\" alt=\"Open In Colab\"/></a>"
]
},
{
"cell_type": "markdown",
"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!"
],
"metadata": {
"id": "AuW-xg_bTsaF"
},
"id": "AuW-xg_bTsaF"
},
{
"cell_type": "code",
"execution_count": 1,
"source": [
"import os\n",
"import zipfile\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"
],
"outputs": [],
"metadata": {
"id": "dn-6c02VmqiN"
},
"id": "dn-6c02VmqiN"
},
{
"cell_type": "markdown",
"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."
],
"metadata": {
"id": "bLTQd84RUs1j"
},
"id": "bLTQd84RUs1j"
},
{
"cell_type": "code",
"execution_count": 2,
"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_3367a.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()"
],
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"--2022-02-15 07:31:10-- https://download.microsoft.com/download/3/E/1/3E1C3F21-ECDB-4869-8368-6DEBA77B919F/kagglecatsanddogs_3367a.zip\n",
"Resolving download.microsoft.com (download.microsoft.com)... 23.78.12.116, 2600:1407:3c00:108c::e59, 2600:1407:3c00:10a2::e59\n",
"Connecting to download.microsoft.com (download.microsoft.com)|23.78.12.116|:443... connected.\n",
"HTTP request sent, awaiting response... 200 OK\n",
"Length: 824894548 (787M) [application/octet-stream]\n",
"Saving to: ‘/tmp/cats-and-dogs.zip’\n",
"\n",
"/tmp/cats-and-dogs. 100%[===================>] 786.68M 51.9MB/s in 24s \n",
"\n",
"2022-02-15 07:31:35 (32.2 MB/s) - ‘/tmp/cats-and-dogs.zip’ saved [824894548/824894548]\n",
"\n"
]
}
],
"metadata": {
"id": "3sd9dQWa23aj",
"lines_to_next_cell": 2,
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "4167f4bb-fe04-4a30-fc29-f408c3c6d342"
},
"id": "3sd9dQWa23aj"
},
{
"cell_type": "markdown",
"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."
],
"metadata": {
"id": "e_HsUV9WVJHL"
},
"id": "e_HsUV9WVJHL"
},
{
"cell_type": "code",
"execution_count": 3,
"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",
"\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.\")"
],
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"There are 12501 images of dogs.\n",
"There are 12501 images of cats.\n"
]
}
],
"metadata": {
"id": "DM851ZmN28J3",
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "ed45a34e-ad39-42dc-cd18-b25d3600ea97"
},
"id": "DM851ZmN28J3"
},
{
"cell_type": "markdown",
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"There are 12501 images of dogs.\n",
"There are 12501 images of cats.\n",
"```"
],
"metadata": {
"id": "G7dI86rmRGmC"
},
"id": "G7dI86rmRGmC"
},
{
"cell_type": "markdown",
"source": [
"You will need a directory for cats-v-dogs, and subdirectories for training\n",
"and testing. These in turn will need subdirectories for 'cats' and 'dogs'. To accomplish this, complete the `create_train_test_dirs` below:"
],
"metadata": {
"id": "iFbMliudNIjW"
},
"id": "iFbMliudNIjW"
},
{
"cell_type": "code",
"execution_count": 4,
"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_test_dirs\n",
"def create_train_test_dirs(root_path):\n",
" ### START CODE HERE\n",
" path = os.path.join(root_path, \"training\")\n",
" os.makedirs(path)\n",
" path_1 = os.path.join(path, \"cats\")\n",
" os.makedirs(path_1)\n",
" path_2 = os.path.join(path, \"dogs\")\n",
" os.makedirs(path_2)\n",
" path = os.path.join(root_path, \"testing\")\n",
" os.makedirs(path)\n",
" path_3 = os.path.join(path, \"cats\")\n",
" os.makedirs(path_3)\n",
" path_4 = os.path.join(path, \"dogs\")\n",
" os.makedirs(path_4)\n",
" # HINT:\n",
" # Use os.makedirs to create your directories with intermediate subdirectories\n",
"\n",
" \n",
"\n",
" ### END CODE HERE\n",
"\n",
" \n",
"try:\n",
" create_train_test_dirs(root_path=root_dir)\n",
"except FileExistsError:\n",
" print(\"You should not be seeing this since the upper directory is removed beforehand\")"
],
"outputs": [],
"metadata": {
"cellView": "code",
"id": "F-QkLjxpmyK2"
},
"id": "F-QkLjxpmyK2"
},
{
"cell_type": "code",
"execution_count": 5,
"source": [
"# Test your create_train_test_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))"
],
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"/tmp/cats-v-dogs/testing\n",
"/tmp/cats-v-dogs/training\n",
"/tmp/cats-v-dogs/testing/cats\n",
"/tmp/cats-v-dogs/testing/dogs\n",
"/tmp/cats-v-dogs/training/cats\n",
"/tmp/cats-v-dogs/training/dogs\n"
]
}
],
"metadata": {
"id": "5dhtL344OK00",
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "8556884f-07ec-41a4-87b7-ea1a346c1159"
},
"id": "5dhtL344OK00"
},
{
"cell_type": "markdown",
"source": [
"**Expected Output (directory order might vary):**\n",
"\n",
"``` txt\n",
"/tmp/cats-v-dogs/training\n",
"/tmp/cats-v-dogs/testing\n",
"/tmp/cats-v-dogs/training/cats\n",
"/tmp/cats-v-dogs/training/dogs\n",
"/tmp/cats-v-dogs/testing/cats\n",
"/tmp/cats-v-dogs/testing/dogs\n",
"\n",
"```"
],
"metadata": {
"id": "D7A0RK3IQsvg"
},
"id": "D7A0RK3IQsvg"
},
{
"cell_type": "markdown",
"source": [
"Code the `split_data` function which takes in the following arguments:\n",
"- SOURCE: directory containing the files\n",
"\n",
"- TRAINING: directory that a portion of the files will be copied to (will be used for training)\n",
"- TESTING: directory that a portion of the files will be copied to (will be used for testing)\n",
"- SPLIT SIZE: to determine the portion\n",
"\n",
"The files should be randomized, so that the training set is a random sample of the files, and the test set is made up of the remaining files.\n",
"\n",
"For example, if `SOURCE` is `PetImages/Cat`, and `SPLIT` SIZE is .9 then 90% of the images in `PetImages/Cat` will be copied to the `TRAINING` dir\n",
"and 10% of the images will be copied to the `TESTING` dir.\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"
],
"metadata": {
"id": "R93T7HdE5txZ"
},
"id": "R93T7HdE5txZ"
},
{
"cell_type": "code",
"execution_count": 6,
"source": [
"# GRADED FUNCTION: split_data\n",
"def split_data(SOURCE, TRAINING, TESTING, SPLIT_SIZE):\n",
"\n",
" ### START CODE HERE\n",
" files = []\n",
" for filename in os.listdir(SOURCE):\n",
" file = SOURCE + filename\n",
" if os.path.getsize(file) > 0:\n",
" files.append(filename)\n",
" else:\n",
" print(filename + ' is zero length, so ignoring.')\n",
"\n",
" training_length = int(len(files) * SPLIT_SIZE)\n",
" testing_length = int(len(files) - training_length)\n",
" shuffled_set = random.sample(files, len(files))\n",
" training_set = shuffled_set[0:training_length]\n",
" testing_set = shuffled_set[-testing_length:]\n",
" \n",
" for filename in training_set:\n",
" src_file = SOURCE + filename\n",
" dest_file = TRAINING + filename\n",
" copyfile(src_file, dest_file)\n",
" \n",
" for filename in testing_set:\n",
" src_file = SOURCE + filename\n",
" dest_file = TESTING + filename\n",
" copyfile(src_file, dest_file)\n",
"\n",
" pass\n",
"\n",
"\n",
" ### END CODE HERE\n"
],
"outputs": [],
"metadata": {
"cellView": "code",
"id": "zvSODo0f9LaU"
},
"id": "zvSODo0f9LaU"
},
{
"cell_type": "code",
"execution_count": 7,
"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",
"TESTING_DIR = \"/tmp/cats-v-dogs/testing/\"\n",
"\n",
"TRAINING_CATS_DIR = os.path.join(TRAINING_DIR, \"cats/\")\n",
"TESTING_CATS_DIR = os.path.join(TESTING_DIR, \"cats/\")\n",
"\n",
"TRAINING_DOGS_DIR = os.path.join(TRAINING_DIR, \"dogs/\")\n",
"TESTING_DOGS_DIR = os.path.join(TESTING_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(TESTING_CATS_DIR)) > 0:\n",
" for file in os.scandir(TESTING_CATS_DIR):\n",
" os.remove(file.path)\n",
"if len(os.listdir(TESTING_DOGS_DIR)) > 0:\n",
" for file in os.scandir(TESTING_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, TESTING_CATS_DIR, split_size)\n",
"split_data(DOG_SOURCE_DIR, TRAINING_DOGS_DIR, TESTING_DOGS_DIR, split_size)\n",
"\n",
"# Check that the number of images matches the expected output\n",
"print(f\"\\n\\nThere 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(TESTING_CATS_DIR))} images of cats for testing\")\n",
"print(f\"There are {len(os.listdir(TESTING_DOGS_DIR))} images of dogs for testing\")"
],
"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",
"There are 11250 images of cats for training\n",
"There are 11250 images of dogs for training\n",
"There are 1250 images of cats for testing\n",
"There are 1250 images of dogs for testing\n"
]
}
],
"metadata": {
"id": "FlIdoUeX9S-9",
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "ca5f943f-da58-43ed-c231-45e7e0ae23fc"
},
"id": "FlIdoUeX9S-9"
},
{
"cell_type": "markdown",
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"666.jpg is zero length, so ignoring.\n",
"11702.jpg is zero length, so ignoring.\n",
"```\n",
"\n",
"```\n",
"There are 11250 images of cats for training\n",
"There are 11250 images of dogs for training\n",
"There are 1250 images of cats for testing\n",
"There are 1250 images of dogs for testing\n",
"```"
],
"metadata": {
"id": "hvskJNOFVSaz"
},
"id": "hvskJNOFVSaz"
},
{
"cell_type": "markdown",
"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",
"**Note:** So far, you have seen the term `testing` being used a lot for referring to a subset of images within the dataset. In this exercise, all of the `testing` data is actually being used as `validation` data. This is not very important within the context of the task at hand but it is worth mentioning to avoid confusion."
],
"metadata": {
"id": "Zil4QmOD_mXF"
},
"id": "Zil4QmOD_mXF"
},
{
"cell_type": "code",
"execution_count": null,
"source": [
"# GRADED FUNCTION: train_val_generators\n",
"def train_val_generators(TRAINING_DIR, VALIDATION_DIR):\n",
" ### START CODE HERE\n",
"\n",
" # Instantiate the ImageDataGenerator class (don't forget to set the rescale argument)\n",
" train_datagen = ImageDataGenerator(rescale=1.0/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=100,\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.0/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=100,\n",
" class_mode='binary',\n",
" target_size=(150, 150))\n",
" ### END CODE HERE\n",
" return train_generator, validation_generator\n"
],
"outputs": [],
"metadata": {
"cellView": "code",
"id": "fQrZfVgz4j2g"
},
"id": "fQrZfVgz4j2g"
},
{
"cell_type": "code",
"execution_count": null,
"source": [
"# Test your generators\n",
"train_generator, validation_generator = train_val_generators(TRAINING_DIR, TESTING_DIR)"
],
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"Found 22498 images belonging to 2 classes.\n",
"Found 2500 images belonging to 2 classes.\n"
]
}
],
"metadata": {
"id": "qM7FxrjGiobD",
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "e606a91b-9ee2-401f-adfa-2ccd703a2f85"
},
"id": "qM7FxrjGiobD"
},
{
"cell_type": "markdown",
"source": [
"**Expected Output:**\n",
"\n",
"```\n",
"Found 22498 images belonging to 2 classes.\n",
"Found 2500 images belonging to 2 classes.\n",
"```\n"
],
"metadata": {
"id": "tiPNmSfZjHwJ"
},
"id": "tiPNmSfZjHwJ"
},
{
"cell_type": "markdown",
"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.**"
],
"metadata": {
"id": "TI3oEmyQCZoO"
},
"id": "TI3oEmyQCZoO"
},
{
"cell_type": "code",
"execution_count": null,
"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",
" tf.keras.layers.Conv2D(16,(3,3), activation = 'relu', input_shape=(150,150,3)),\n",
" tf.keras.layers.MaxPooling2D(2,2),\n",
" tf.keras.layers.Conv2D(32,(3,3), activation = 'relu'),\n",
" tf.keras.layers.MaxPooling2D(2,2),\n",
" tf.keras.layers.Conv2D(64,(3,3), activation = 'relu'),\n",
" tf.keras.layers.MaxPooling2D(2,2),\n",
" tf.keras.layers.Flatten(),\n",
" tf.keras.layers.Dense(512, activation = 'relu'),\n",
" tf.keras.layers.Dense(1, activation='sigmoid')\n",
" ])\n",
"\n",
" \n",
" model.compile(optimizer = tf.keras.optimizers.RMSprop(learning_rate=0.001),\n",
" loss='binary_crossentropy',\n",
" metrics=['accuracy']) \n",
" \n",
" ### END CODE HERE\n",
"\n",
" return model\n"
],
"outputs": [],
"metadata": {
"cellView": "code",
"id": "oDPK8tUB_O9e",
"lines_to_next_cell": 2
},
"id": "oDPK8tUB_O9e"
},
{
"cell_type": "markdown",
"source": [
"Now it is time to train your model!\n",
"\n",
"**Note:** You can ignore the `UserWarning: Possibly corrupt EXIF data.` warnings."
],
"metadata": {
"id": "SMFNJZmTCZv6"
},
"id": "SMFNJZmTCZv6"
},
{
"cell_type": "code",
"execution_count": null,
"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)"
],
"outputs": [
{
"output_type": "stream",
"name": "stdout",
"text": [
"Epoch 1/15\n",
" 34/225 [===>..........................] - ETA: 1:03 - loss: 1.0842 - accuracy: 0.5456"
]
},
{
"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": [
"225/225 [==============================] - 96s 382ms/step - loss: 0.6908 - accuracy: 0.6427 - val_loss: 0.5351 - val_accuracy: 0.7416\n",
"Epoch 2/15\n",
"225/225 [==============================] - 86s 382ms/step - loss: 0.5124 - accuracy: 0.7456 - val_loss: 0.4600 - val_accuracy: 0.7768\n",
"Epoch 3/15\n",
"225/225 [==============================] - 85s 378ms/step - loss: 0.4453 - accuracy: 0.7908 - val_loss: 0.4301 - val_accuracy: 0.8072\n",
"Epoch 4/15\n",
"225/225 [==============================] - 86s 382ms/step - loss: 0.3894 - accuracy: 0.8236 - val_loss: 0.4373 - val_accuracy: 0.7940\n",
"Epoch 5/15\n",
"225/225 [==============================] - 85s 378ms/step - loss: 0.3326 - accuracy: 0.8523 - val_loss: 0.4453 - val_accuracy: 0.8048\n",
"Epoch 6/15\n",
"225/225 [==============================] - 85s 378ms/step - loss: 0.2727 - accuracy: 0.8858 - val_loss: 0.4375 - val_accuracy: 0.8140\n",
"Epoch 7/15\n",
"225/225 [==============================] - 85s 380ms/step - loss: 0.2021 - accuracy: 0.9162 - val_loss: 0.4901 - val_accuracy: 0.8252\n",
"Epoch 8/15\n",
"225/225 [==============================] - 87s 384ms/step - loss: 0.1393 - accuracy: 0.9448 - val_loss: 0.4805 - val_accuracy: 0.8308\n",
"Epoch 9/15\n",
"225/225 [==============================] - 88s 392ms/step - loss: 0.0939 - accuracy: 0.9661 - val_loss: 0.6171 - val_accuracy: 0.8332\n",
"Epoch 10/15\n",
"225/225 [==============================] - 89s 397ms/step - loss: 0.0635 - accuracy: 0.9788 - val_loss: 0.5303 - val_accuracy: 0.8296\n",
"Epoch 11/15\n",
"225/225 [==============================] - 89s 394ms/step - loss: 0.0482 - accuracy: 0.9844 - val_loss: 0.8566 - val_accuracy: 0.8396\n",
"Epoch 12/15\n",
"225/225 [==============================] - 88s 391ms/step - loss: 0.0543 - accuracy: 0.9845 - val_loss: 0.8273 - val_accuracy: 0.8316\n",
"Epoch 13/15\n",
"225/225 [==============================] - 88s 391ms/step - loss: 0.0397 - accuracy: 0.9879 - val_loss: 0.9041 - val_accuracy: 0.8328\n",
"Epoch 14/15\n",
"225/225 [==============================] - 88s 392ms/step - loss: 0.0541 - accuracy: 0.9867 - val_loss: 1.1718 - val_accuracy: 0.8276\n",
"Epoch 15/15\n",
"225/225 [==============================] - 89s 394ms/step - loss: 0.0374 - accuracy: 0.9905 - val_loss: 1.3956 - val_accuracy: 0.8288\n"
]
}
],
"metadata": {
"id": "5qE1G6JB4fMn",
"colab": {
"base_uri": "https://localhost:8080/"
},
"outputId": "901a4065-bfff-45ed-d4ad-cd39e035730b"
},
"id": "5qE1G6JB4fMn"
},
{
"cell_type": "markdown",
"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."
],
"metadata": {
"id": "VGsaDMc-GMd4"
},
"id": "VGsaDMc-GMd4"
},
{
"cell_type": "code",
"execution_count": null,
"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()"
],
"outputs": [
{
"output_type": "display_data",
"data": {
"image/png": "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",
"text/plain": [
"<Figure size 432x288 with 1 Axes>"
]
},
"metadata": {
"needs_background": "light"
}
},
{
"output_type": "stream",
"name": "stdout",
"text": [
"\n"
]
},
{
"output_type": "display_data",
"data": {
"image/png": "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",
"text/plain": [
"<Figure size 432x288 with 1 Axes>"
]
},
"metadata": {
"needs_background": "light"
}
}
],
"metadata": {
"id": "MWZrJN4-65RC",
"colab": {
"base_uri": "https://localhost:8080/",
"height": 546
},
"outputId": "2ccfc80b-9963-40b7-d30c-adffc58fc849"
},
"id": "MWZrJN4-65RC"
},
{
"cell_type": "markdown",
"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:"
],
"metadata": {
"id": "NYIaqsN2pav6"
},
"id": "NYIaqsN2pav6"
},
{
"cell_type": "code",
"execution_count": null,
"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()"
],
"outputs": [
{
"output_type": "display_data",
"data": {
"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 ",
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {}
},
{
"output_type": "display_data",
"data": {
"application/javascript": "download(\"download_e05db2ba-3685-4b7d-9fb7-a1152af6516f\", \"history.pkl\", 628)",
"text/plain": [
"<IPython.core.display.Javascript object>"
]
},
"metadata": {}
}
],
"metadata": {
"id": "yWcrc9nZTsHj",
"colab": {
"base_uri": "https://localhost:8080/",
"height": 17
},
"outputId": "18e8a00d-369f-4cad-8ff3-da3654f5d208"
},
"id": "yWcrc9nZTsHj"
},
{
"cell_type": "markdown",
"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."
],
"metadata": {
"id": "ycjhQNn4cQmU"
},
"id": "ycjhQNn4cQmU"
},
{
"cell_type": "markdown",
"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!**"
],
"metadata": {
"id": "joAaZSWWpbOI"
},
"id": "joAaZSWWpbOI"
}
],
"metadata": {
"accelerator": "GPU",
"kernelspec": {
"name": "python3",
"display_name": "Python 3.9.5 64-bit"
},
"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.9.5"
},
"colab": {
"name": "submissionC2W1.ipynb",
"provenance": [],
"include_colab_link": true
},
"interpreter": {
"hash": "de21584f6befcc045abdaaa1de6535ec0aed54614e7768465545e09c4ac2dac2"
}
},
"nbformat": 4,
"nbformat_minor": 5
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment