From 146e6fc062826979ff9ca144dc0aa973e230c46e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Aur=C3=A9lien=20Geron?= Date: Wed, 8 Dec 2021 11:58:36 +1300 Subject: [PATCH] Big update of chapter 10 for 3rd edition --- 10_neural_nets_with_keras.ipynb | 1500 ++++++++++++++++++------------- 1 file changed, 896 insertions(+), 604 deletions(-) diff --git a/10_neural_nets_with_keras.ipynb b/10_neural_nets_with_keras.ipynb index aaf8c4b..35bc446 100644 --- a/10_neural_nets_with_keras.ipynb +++ b/10_neural_nets_with_keras.ipynb @@ -4,14 +4,14 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "**Chapter 9 – Introduction to Artificial Neural Networks with Keras**" + "**Chapter 10 – Introduction to Artificial Neural Networks with Keras**" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "_This notebook contains all the sample code and solutions to the exercises in chapter 9._" + "_This notebook contains all the sample code and solutions to the exercises in chapter 10._" ] }, { @@ -20,17 +20,19 @@ "source": [ "\n", " \n", " \n", "
\n", - " \"Open\n", + " \"Open\n", " \n", - " \n", + " \n", "
" ] }, { "cell_type": "markdown", - "metadata": {}, + "metadata": { + "tags": [] + }, "source": [ "# Setup" ] @@ -39,7 +41,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "First, let's import a few common modules, ensure MatplotLib plots figures inline and prepare a function to save the figures." + "This project requires Python 3.8 or above:" ] }, { @@ -48,35 +50,84 @@ "metadata": {}, "outputs": [], "source": [ - "# Python ≥3.8 is required\n", "import sys\n", - "assert sys.version_info >= (3, 8)\n", "\n", - "# Common imports\n", - "import numpy as np\n", + "assert sys.version_info >= (3, 8)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "It also requires Scikit-Learn ≥ 1.0.1:" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [], + "source": [ + "import sklearn\n", + "\n", + "assert sklearn.__version__ >= \"1.0.1\"" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "And TensorFlow ≥ 2.6:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [], + "source": [ + "import tensorflow as tf\n", + "\n", + "assert tf.__version__ >= \"2.6.0\"" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "As we did in previous chapters, let's define the default font sizes to make the figures prettier:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [], + "source": [ + "import matplotlib.pyplot as plt\n", + "\n", + "plt.rc('font', size=14)\n", + "plt.rc('axes', labelsize=14, titlesize=14)\n", + "plt.rc('legend', fontsize=14)\n", + "plt.rc('xtick', labelsize=10)\n", + "plt.rc('ytick', labelsize=10)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "And let's create the `images/ann` folder (if it doesn't already exist), and define the `save_fig()` function which is used through this notebook to save the figures in high-res for the book:" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [], + "source": [ "from pathlib import Path\n", "\n", - "# Scikit-Learn ≥1.0 is required\n", - "import sklearn\n", - "assert sklearn.__version__ >= \"1.0\"\n", - "\n", - "# TensorFlow ≥2.6 is required\n", - "import tensorflow as tf\n", - "assert tf.__version__ >= \"2.6\"\n", - "\n", - "# to make this notebook's output stable across runs\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)\n", - "\n", - "# To plot pretty figures\n", - "%matplotlib inline\n", - "import matplotlib as mpl\n", - "import matplotlib.pyplot as plt\n", - "mpl.rc('axes', labelsize=14)\n", - "mpl.rc('xtick', labelsize=12)\n", - "mpl.rc('ytick', labelsize=12)\n", - "\n", - "# Where to save the figures\n", "IMAGES_PATH = Path() / \"images\" / \"ann\"\n", "IMAGES_PATH.mkdir(parents=True, exist_ok=True)\n", "\n", @@ -91,103 +142,15 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# Perceptrons" + "First, let's import a few common modules, ensure MatplotLib plots figures inline and prepare a function to save the figures." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "**Note**: we set `max_iter` and `tol` explicitly to avoid warnings about the fact that their default value will change in future versions of Scikit-Learn." - ] - }, - { - "cell_type": "code", - "execution_count": 2, - "metadata": {}, - "outputs": [], - "source": [ - "import numpy as np\n", - "from sklearn.datasets import load_iris\n", - "from sklearn.linear_model import Perceptron\n", - "\n", - "iris = load_iris()\n", - "X = iris.data[:, (2, 3)] # petal length, petal width\n", - "y = (iris.target == 0).astype(np.int)\n", - "\n", - "per_clf = Perceptron(max_iter=1000, tol=1e-3, random_state=42)\n", - "per_clf.fit(X, y)\n", - "\n", - "y_pred = per_clf.predict([[2, 0.5]])" - ] - }, - { - "cell_type": "code", - "execution_count": 3, - "metadata": {}, - "outputs": [], - "source": [ - "y_pred" - ] - }, - { - "cell_type": "code", - "execution_count": 4, - "metadata": {}, - "outputs": [], - "source": [ - "a = -per_clf.coef_[0][0] / per_clf.coef_[0][1]\n", - "b = -per_clf.intercept_ / per_clf.coef_[0][1]\n", - "\n", - "axes = [0, 5, 0, 2]\n", - "\n", - "x0, x1 = np.meshgrid(\n", - " np.linspace(axes[0], axes[1], 500).reshape(-1, 1),\n", - " np.linspace(axes[2], axes[3], 200).reshape(-1, 1),\n", - " )\n", - "X_new = np.c_[x0.ravel(), x1.ravel()]\n", - "y_predict = per_clf.predict(X_new)\n", - "zz = y_predict.reshape(x0.shape)\n", - "\n", - "plt.figure(figsize=(10, 4))\n", - "plt.plot(X[y==0, 0], X[y==0, 1], \"bs\", label=\"Not Iris-Setosa\")\n", - "plt.plot(X[y==1, 0], X[y==1, 1], \"yo\", label=\"Iris-Setosa\")\n", - "\n", - "plt.plot([axes[0], axes[1]], [a * axes[0] + b, a * axes[1] + b], \"k-\", linewidth=3)\n", - "from matplotlib.colors import ListedColormap\n", - "custom_cmap = ListedColormap(['#9898ff', '#fafab0'])\n", - "\n", - "plt.contourf(x0, x1, zz, cmap=custom_cmap)\n", - "plt.xlabel(\"Petal length\", fontsize=14)\n", - "plt.ylabel(\"Petal width\", fontsize=14)\n", - "plt.legend(loc=\"lower right\", fontsize=14)\n", - "plt.axis(axes)\n", - "\n", - "save_fig(\"perceptron_iris_plot\")\n", - "plt.show()" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# Activation functions" - ] - }, - { - "cell_type": "code", - "execution_count": 5, - "metadata": {}, - "outputs": [], - "source": [ - "def sigmoid(z):\n", - " return 1 / (1 + np.exp(-z))\n", - "\n", - "def relu(z):\n", - " return np.maximum(0, z)\n", - "\n", - "def derivative(f, z, eps=0.000001):\n", - " return (f(z + eps) - f(z - eps))/(2 * eps)" + "# From Biological to Artificial Neurons\n", + "## The Perceptron" ] }, { @@ -196,34 +159,19 @@ "metadata": {}, "outputs": [], "source": [ - "z = np.linspace(-5, 5, 200)\n", + "import numpy as np\n", + "from sklearn.datasets import load_iris\n", + "from sklearn.linear_model import Perceptron\n", "\n", - "plt.figure(figsize=(11,4))\n", + "iris = load_iris(as_frame=True)\n", + "X = iris.data[[\"petal length (cm)\", \"petal width (cm)\"]].values\n", + "y = (iris.target == 0) # Iris setosa\n", "\n", - "plt.subplot(121)\n", - "plt.plot(z, np.sign(z), \"r-\", linewidth=1, label=\"Step\")\n", - "plt.plot(z, sigmoid(z), \"g--\", linewidth=2, label=\"Sigmoid\")\n", - "plt.plot(z, np.tanh(z), \"b-\", linewidth=2, label=\"Tanh\")\n", - "plt.plot(z, relu(z), \"m-.\", linewidth=2, label=\"ReLU\")\n", - "plt.grid(True)\n", - "plt.legend(loc=\"center right\", fontsize=14)\n", - "plt.title(\"Activation functions\", fontsize=14)\n", - "plt.axis([-5, 5, -1.2, 1.2])\n", + "per_clf = Perceptron(random_state=42)\n", + "per_clf.fit(X, y)\n", "\n", - "plt.subplot(122)\n", - "plt.plot(z, derivative(np.sign, z), \"r-\", linewidth=1, label=\"Step\")\n", - "plt.plot(0, 0, \"ro\", markersize=5)\n", - "plt.plot(0, 0, \"rx\", markersize=10)\n", - "plt.plot(z, derivative(sigmoid, z), \"g--\", linewidth=2, label=\"Sigmoid\")\n", - "plt.plot(z, derivative(np.tanh, z), \"b-\", linewidth=2, label=\"Tanh\")\n", - "plt.plot(z, derivative(relu, z), \"m-.\", linewidth=2, label=\"ReLU\")\n", - "plt.grid(True)\n", - "#plt.legend(loc=\"center right\", fontsize=14)\n", - "plt.title(\"Derivatives\", fontsize=14)\n", - "plt.axis([-5, 5, -0.2, 1.2])\n", - "\n", - "save_fig(\"activation_functions_plot\")\n", - "plt.show()" + "X_new = [[2, 0.5], [3, 1]]\n", + "y_pred = per_clf.predict(X_new) # predicts True and False for these 2 flowers" ] }, { @@ -232,57 +180,38 @@ "metadata": {}, "outputs": [], "source": [ - "def heaviside(z):\n", - " return (z >= 0).astype(z.dtype)\n", - "\n", - "def mlp_xor(x1, x2, activation=heaviside):\n", - " return activation(-activation(x1 + x2 - 1.5) + activation(x1 + x2 - 0.5) - 0.5)" + "y_pred" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The `Perceptron` is equivalent to a `SGDClassifier` with `loss=\"perceptron\"`, no regularization, and a constant learning rate equal to 1:" ] }, { "cell_type": "code", "execution_count": 8, - "metadata": { - "scrolled": true - }, + "metadata": {}, "outputs": [], "source": [ - "x1s = np.linspace(-0.2, 1.2, 100)\n", - "x2s = np.linspace(-0.2, 1.2, 100)\n", - "x1, x2 = np.meshgrid(x1s, x2s)\n", + "# not in the book – extra material\n", "\n", - "z1 = mlp_xor(x1, x2, activation=heaviside)\n", - "z2 = mlp_xor(x1, x2, activation=sigmoid)\n", + "from sklearn.linear_model import SGDClassifier\n", "\n", - "plt.figure(figsize=(10,4))\n", - "\n", - "plt.subplot(121)\n", - "plt.contourf(x1, x2, z1)\n", - "plt.plot([0, 1], [0, 1], \"gs\", markersize=20)\n", - "plt.plot([0, 1], [1, 0], \"y^\", markersize=20)\n", - "plt.title(\"Activation function: heaviside\", fontsize=14)\n", - "plt.grid(True)\n", - "\n", - "plt.subplot(122)\n", - "plt.contourf(x1, x2, z2)\n", - "plt.plot([0, 1], [0, 1], \"gs\", markersize=20)\n", - "plt.plot([0, 1], [1, 0], \"y^\", markersize=20)\n", - "plt.title(\"Activation function: sigmoid\", fontsize=14)\n", - "plt.grid(True)" + "sgd_clf = SGDClassifier(loss=\"perceptron\", penalty=None,\n", + " learning_rate=\"constant\", eta0=1, random_state=42)\n", + "sgd_clf.fit(X, y)\n", + "assert (sgd_clf.coef_ == per_clf.coef_).all()\n", + "assert (sgd_clf.intercept_ == per_clf.intercept_).all()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "# Building an Image Classifier" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "First let's import TensorFlow and Keras." + "When the Perceptron finds a decision boundary that properly separates the classes, it stops learning. This means that the decision boundary is often quite close to one class:" ] }, { @@ -291,8 +220,41 @@ "metadata": {}, "outputs": [], "source": [ - "import tensorflow as tf\n", - "from tensorflow import keras" + "# not in the book – extra material\n", + "\n", + "import matplotlib.pyplot as plt\n", + "from matplotlib.colors import ListedColormap\n", + "\n", + "a = -per_clf.coef_[0, 0] / per_clf.coef_[0, 1]\n", + "b = -per_clf.intercept_ / per_clf.coef_[0, 1]\n", + "axes = [0, 5, 0, 2]\n", + "x0, x1 = np.meshgrid(\n", + " np.linspace(axes[0], axes[1], 500).reshape(-1, 1),\n", + " np.linspace(axes[2], axes[3], 200).reshape(-1, 1),\n", + ")\n", + "X_new = np.c_[x0.ravel(), x1.ravel()]\n", + "y_predict = per_clf.predict(X_new)\n", + "zz = y_predict.reshape(x0.shape)\n", + "custom_cmap = ListedColormap(['#9898ff', '#fafab0'])\n", + "\n", + "plt.figure(figsize=(7, 3))\n", + "plt.plot(X[y == 0, 0], X[y == 0, 1], \"bs\", label=\"Not Iris setosa\")\n", + "plt.plot(X[y == 1, 0], X[y == 1, 1], \"yo\", label=\"Iris setosa\")\n", + "plt.plot([axes[0], axes[1]], [a * axes[0] + b, a * axes[1] + b], \"k-\",\n", + " linewidth=3)\n", + "plt.contourf(x0, x1, zz, cmap=custom_cmap)\n", + "plt.xlabel(\"Petal length\")\n", + "plt.ylabel(\"Petal width\")\n", + "plt.legend(loc=\"lower right\")\n", + "plt.axis(axes)\n", + "plt.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "**Activation functions**" ] }, { @@ -301,7 +263,58 @@ "metadata": {}, "outputs": [], "source": [ - "tf.__version__" + "# not in the book – this code generates and saves Figure 10–8\n", + "\n", + "from scipy.special import expit as sigmoid\n", + "\n", + "def relu(z):\n", + " return np.maximum(0, z)\n", + "\n", + "def derivative(f, z, eps=0.000001):\n", + " return (f(z + eps) - f(z - eps))/(2 * eps)\n", + "\n", + "max_z = 4.5\n", + "z = np.linspace(-max_z, max_z, 200)\n", + "\n", + "plt.figure(figsize=(11, 3.1))\n", + "\n", + "plt.subplot(121)\n", + "plt.plot([-max_z, 0], [0, 0], \"r-\", linewidth=2, label=\"Heaviside\")\n", + "plt.plot(z, relu(z), \"m-.\", linewidth=2, label=\"ReLU\")\n", + "plt.plot([0, 0], [0, 1], \"r-\", linewidth=0.5)\n", + "plt.plot([0, max_z], [1, 1], \"r-\", linewidth=2)\n", + "plt.plot(z, sigmoid(z), \"g--\", linewidth=2, label=\"Sigmoid\")\n", + "plt.plot(z, np.tanh(z), \"b-\", linewidth=1, label=\"Tanh\")\n", + "plt.grid(True)\n", + "plt.title(\"Activation functions\")\n", + "plt.axis([-max_z, max_z, -1.65, 2.4])\n", + "plt.gca().set_yticks([-1, 0, 1, 2])\n", + "plt.legend(loc=\"lower right\", fontsize=13)\n", + "\n", + "plt.subplot(122)\n", + "plt.plot(z, derivative(np.sign, z), \"r-\", linewidth=2, label=\"Heaviside\")\n", + "plt.plot(0, 0, \"ro\", markersize=5)\n", + "plt.plot(0, 0, \"rx\", markersize=10)\n", + "plt.plot(z, derivative(sigmoid, z), \"g--\", linewidth=2, label=\"Sigmoid\")\n", + "plt.plot(z, derivative(np.tanh, z), \"b-\", linewidth=1, label=\"Tanh\")\n", + "plt.plot([-max_z, 0], [0, 0], \"m-.\", linewidth=2)\n", + "plt.plot([0, max_z], [1, 1], \"m-.\", linewidth=2)\n", + "plt.plot([0, 0], [0, 1], \"m-.\", linewidth=1.2)\n", + "plt.plot(0, 1, \"mo\", markersize=5)\n", + "plt.plot(0, 1, \"mx\", markersize=10)\n", + "plt.grid(True)\n", + "plt.title(\"Derivatives\")\n", + "plt.axis([-max_z, max_z, -0.2, 1.2])\n", + "\n", + "save_fig(\"activation_functions_plot\")\n", + "plt.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Regression MLPs" ] }, { @@ -310,14 +323,24 @@ "metadata": {}, "outputs": [], "source": [ - "tf.keras.__version__" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "Let's start by loading the fashion MNIST dataset. Keras has a number of functions to load popular datasets in `tf.keras.datasets`. The dataset is already split for you between a training set and a test set, but it can be useful to split the training set further to have a validation set:" + "from sklearn.datasets import fetch_california_housing\n", + "from sklearn.metrics import mean_squared_error\n", + "from sklearn.model_selection import train_test_split\n", + "from sklearn.neural_network import MLPRegressor\n", + "from sklearn.pipeline import make_pipeline\n", + "from sklearn.preprocessing import StandardScaler\n", + "\n", + "housing = fetch_california_housing()\n", + "X_train_full, X_test, y_train_full, y_test = train_test_split(\n", + " housing.data, housing.target, random_state=42)\n", + "X_train, X_valid, y_train, y_valid = train_test_split(\n", + " X_train_full, y_train_full, random_state=42)\n", + "\n", + "mlp_reg = MLPRegressor(hidden_layer_sizes=[50, 50, 50], random_state=42)\n", + "pipeline = make_pipeline(StandardScaler(), mlp_reg)\n", + "pipeline.fit(X_train, y_train)\n", + "y_pred = pipeline.predict(X_valid)\n", + "rmse = mean_squared_error(y_valid, y_pred, squared=False)" ] }, { @@ -326,8 +349,70 @@ "metadata": {}, "outputs": [], "source": [ - "fashion_mnist = tf.keras.datasets.fashion_mnist\n", - "(X_train_full, y_train_full), (X_test, y_test) = fashion_mnist.load_data()" + "rmse" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Classification MLPs" + ] + }, + { + "cell_type": "code", + "execution_count": 13, + "metadata": {}, + "outputs": [], + "source": [ + "# not in the book – this was left as an exercise for the reader\n", + "\n", + "from sklearn.datasets import load_iris\n", + "from sklearn.model_selection import train_test_split\n", + "from sklearn.neural_network import MLPClassifier\n", + "\n", + "iris = load_iris()\n", + "X_train_full, X_test, y_train_full, y_test = train_test_split(\n", + " iris.data, iris.target, test_size=0.1, random_state=42)\n", + "X_train, X_valid, y_train, y_valid = train_test_split(\n", + " X_train_full, y_train_full, test_size=0.1, random_state=42)\n", + "\n", + "mlp_clf = MLPClassifier(hidden_layer_sizes=[5], max_iter=10_000,\n", + " random_state=42)\n", + "pipeline = make_pipeline(StandardScaler(), mlp_clf)\n", + "pipeline.fit(X_train, y_train)\n", + "accuracy = pipeline.score(X_valid, y_valid)\n", + "accuracy" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Implementing MLPs with Keras\n", + "## Building an Image Classifier Using the Sequential API\n", + "### Using Keras to load the dataset" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Let's start by loading the fashion MNIST dataset. Keras has a number of functions to load popular datasets in `tf.keras.datasets`. The dataset is already split for you between a training set (60,000 images) and a test set (10,000 images), but it can be useful to split the training set further to have a validation set. We'll use 55,000 images for training, and 5,000 for validation." + ] + }, + { + "cell_type": "code", + "execution_count": 14, + "metadata": {}, + "outputs": [], + "source": [ + "import tensorflow as tf\n", + "\n", + "fashion_mnist = tf.keras.datasets.fashion_mnist.load_data()\n", + "(X_train_full, y_train_full), (X_test, y_test) = fashion_mnist\n", + "X_train, y_train = X_train_full[:-5000], y_train_full[:-5000]\n", + "X_valid, y_valid = X_train_full[-5000:], y_train_full[-5000:]" ] }, { @@ -339,11 +424,11 @@ }, { "cell_type": "code", - "execution_count": 13, + "execution_count": 15, "metadata": {}, "outputs": [], "source": [ - "X_train_full.shape" + "X_train.shape" ] }, { @@ -355,29 +440,27 @@ }, { "cell_type": "code", - "execution_count": 14, + "execution_count": 16, "metadata": {}, "outputs": [], "source": [ - "X_train_full.dtype" + "X_train.dtype" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "Let's split the full training set into a validation set and a (smaller) training set. We also scale the pixel intensities down to the 0-1 range and convert them to floats, by dividing by 255." + "Let's scale the pixel intensities down to the 0-1 range and convert them to floats, by dividing by 255:" ] }, { "cell_type": "code", - "execution_count": 15, + "execution_count": 17, "metadata": {}, "outputs": [], "source": [ - "X_valid, X_train = X_train_full[:5000] / 255., X_train_full[5000:] / 255.\n", - "y_valid, y_train = y_train_full[:5000], y_train_full[5000:]\n", - "X_test = X_test / 255." + "X_train, X_valid, X_test = X_train / 255., X_valid / 255., X_test / 255." ] }, { @@ -390,10 +473,12 @@ }, { "cell_type": "code", - "execution_count": 16, + "execution_count": 18, "metadata": {}, "outputs": [], "source": [ + "# not in the book\n", + "\n", "plt.imshow(X_train[0], cmap=\"binary\")\n", "plt.axis('off')\n", "plt.show()" @@ -408,7 +493,7 @@ }, { "cell_type": "code", - "execution_count": 17, + "execution_count": 19, "metadata": {}, "outputs": [], "source": [ @@ -424,7 +509,7 @@ }, { "cell_type": "code", - "execution_count": 18, + "execution_count": 20, "metadata": {}, "outputs": [], "source": [ @@ -436,32 +521,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "So the first image in the training set is a coat:" - ] - }, - { - "cell_type": "code", - "execution_count": 19, - "metadata": {}, - "outputs": [], - "source": [ - "class_names[y_train[0]]" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "The validation set contains 5,000 images, and the test set contains 10,000 images:" - ] - }, - { - "cell_type": "code", - "execution_count": 20, - "metadata": {}, - "outputs": [], - "source": [ - "X_valid.shape" + "So the first image in the training set is an ankle boot:" ] }, { @@ -470,7 +530,7 @@ "metadata": {}, "outputs": [], "source": [ - "X_test.shape" + "class_names[y_train[0]]" ] }, { @@ -486,6 +546,8 @@ "metadata": {}, "outputs": [], "source": [ + "# not in the book – this cell generates and saves Figure 10–10\n", + "\n", "n_rows = 4\n", "n_cols = 10\n", "plt.figure(figsize=(n_cols * 1.2, n_rows * 1.2))\n", @@ -495,20 +557,30 @@ " plt.subplot(n_rows, n_cols, index + 1)\n", " plt.imshow(X_train[index], cmap=\"binary\", interpolation=\"nearest\")\n", " plt.axis('off')\n", - " plt.title(class_names[y_train[index]], fontsize=12)\n", + " plt.title(class_names[y_train[index]])\n", "plt.subplots_adjust(wspace=0.2, hspace=0.5)\n", - "save_fig('fashion_mnist_plot', tight_layout=False)\n", + "\n", + "save_fig(\"fashion_mnist_plot\")\n", "plt.show()" ] }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Creating the model using the Sequential API" + ] + }, { "cell_type": "code", "execution_count": 23, "metadata": {}, "outputs": [], "source": [ + "tf.random.set_seed(42)\n", "model = tf.keras.Sequential()\n", - "model.add(tf.keras.layers.Flatten(input_shape=[28, 28]))\n", + "model.add(tf.keras.layers.InputLayer(input_shape=[28, 28]))\n", + "model.add(tf.keras.layers.Flatten())\n", "model.add(tf.keras.layers.Dense(300, activation=\"relu\"))\n", "model.add(tf.keras.layers.Dense(100, activation=\"relu\"))\n", "model.add(tf.keras.layers.Dense(10, activation=\"softmax\"))" @@ -520,17 +592,10 @@ "metadata": {}, "outputs": [], "source": [ + "# not in the book – clear the session to reset the name counters\n", "tf.keras.backend.clear_session()\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)" - ] - }, - { - "cell_type": "code", - "execution_count": 25, - "metadata": {}, - "outputs": [], - "source": [ + "tf.random.set_seed(42)\n", + "\n", "model = tf.keras.Sequential([\n", " tf.keras.layers.Flatten(input_shape=[28, 28]),\n", " tf.keras.layers.Dense(300, activation=\"relu\"),\n", @@ -541,16 +606,7 @@ }, { "cell_type": "code", - "execution_count": 26, - "metadata": {}, - "outputs": [], - "source": [ - "model.layers" - ] - }, - { - "cell_type": "code", - "execution_count": 27, + "execution_count": 25, "metadata": {}, "outputs": [], "source": [ @@ -559,16 +615,26 @@ }, { "cell_type": "code", - "execution_count": 28, + "execution_count": 26, "metadata": {}, "outputs": [], "source": [ + "# not in the book – another way to display the model's architecture\n", "tf.keras.utils.plot_model(model, \"my_fashion_mnist_model.png\", show_shapes=True)" ] }, { "cell_type": "code", - "execution_count": 29, + "execution_count": 27, + "metadata": {}, + "outputs": [], + "source": [ + "model.layers" + ] + }, + { + "cell_type": "code", + "execution_count": 28, "metadata": {}, "outputs": [], "source": [ @@ -576,13 +642,23 @@ "hidden1.name" ] }, + { + "cell_type": "code", + "execution_count": 29, + "metadata": {}, + "outputs": [], + "source": [ + "model.get_layer('dense') is hidden1" + ] + }, { "cell_type": "code", "execution_count": 30, "metadata": {}, "outputs": [], "source": [ - "model.get_layer(hidden1.name) is hidden1" + "weights, biases = hidden1.get_weights()\n", + "weights" ] }, { @@ -591,7 +667,7 @@ "metadata": {}, "outputs": [], "source": [ - "weights, biases = hidden1.get_weights()" + "weights.shape" ] }, { @@ -600,7 +676,7 @@ "metadata": {}, "outputs": [], "source": [ - "weights" + "biases" ] }, { @@ -609,7 +685,14 @@ "metadata": {}, "outputs": [], "source": [ - "weights.shape" + "biases.shape" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Compiling the model" ] }, { @@ -617,24 +700,6 @@ "execution_count": 34, "metadata": {}, "outputs": [], - "source": [ - "biases" - ] - }, - { - "cell_type": "code", - "execution_count": 35, - "metadata": {}, - "outputs": [], - "source": [ - "biases.shape" - ] - }, - { - "cell_type": "code", - "execution_count": 36, - "metadata": {}, - "outputs": [], "source": [ "model.compile(loss=\"sparse_categorical_crossentropy\",\n", " optimizer=\"sgd\",\n", @@ -648,15 +713,33 @@ "This is equivalent to:" ] }, + { + "cell_type": "code", + "execution_count": 35, + "metadata": {}, + "outputs": [], + "source": [ + "# not in the book – this cell is equivalent to the previous cell\n", + "model.compile(loss=tf.keras.losses.sparse_categorical_crossentropy,\n", + " optimizer=tf.keras.optimizers.SGD(),\n", + " metrics=[tf.keras.metrics.sparse_categorical_accuracy])" + ] + }, + { + "cell_type": "code", + "execution_count": 36, + "metadata": {}, + "outputs": [], + "source": [ + "# not in the book – this code shows how to convert class ids to one-hot vectors\n", + "tf.keras.utils.to_categorical([0, 5, 1, 0], num_classes=10)" + ] + }, { "cell_type": "markdown", "metadata": {}, "source": [ - "```python\n", - "model.compile(loss=tf.keras.losses.sparse_categorical_crossentropy,\n", - " optimizer=tf.keras.optimizers.SGD(),\n", - " metrics=[tf.keras.metrics.sparse_categorical_accuracy])\n", - "```" + "Note: it's important to set `num_classes` when the number of classes is greater than the maximum class id in the sample." ] }, { @@ -664,6 +747,29 @@ "execution_count": 37, "metadata": {}, "outputs": [], + "source": [ + "# not in the book – this code shows how to convert one-hot vectors to class ids\n", + "np.argmax(\n", + " [[1., 0., 0., 0., 0., 0., 0., 0., 0., 0.],\n", + " [0., 0., 0., 0., 0., 1., 0., 0., 0., 0.],\n", + " [0., 1., 0., 0., 0., 0., 0., 0., 0., 0.],\n", + " [1., 0., 0., 0., 0., 0., 0., 0., 0., 0.]],\n", + " axis=1\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Training and evaluating the model" + ] + }, + { + "cell_type": "code", + "execution_count": 38, + "metadata": {}, + "outputs": [], "source": [ "history = model.fit(X_train, y_train, epochs=30,\n", " validation_data=(X_valid, y_valid))" @@ -671,7 +777,7 @@ }, { "cell_type": "code", - "execution_count": 38, + "execution_count": 39, "metadata": {}, "outputs": [], "source": [ @@ -680,34 +786,27 @@ }, { "cell_type": "code", - "execution_count": 39, + "execution_count": 40, "metadata": {}, "outputs": [], "source": [ "print(history.epoch)" ] }, - { - "cell_type": "code", - "execution_count": 40, - "metadata": {}, - "outputs": [], - "source": [ - "history.history.keys()" - ] - }, { "cell_type": "code", "execution_count": 41, "metadata": {}, "outputs": [], "source": [ + "import matplotlib.pyplot as plt\n", "import pandas as pd\n", "\n", - "pd.DataFrame(history.history).plot(figsize=(8, 5))\n", - "plt.grid(True)\n", - "plt.gca().set_ylim(0, 1)\n", - "save_fig(\"keras_learning_curves_plot\")\n", + "pd.DataFrame(history.history).plot(\n", + " figsize=(8, 5), xlim=[0, 29], ylim=[0, 1], grid=True, xlabel=\"Epoch\",\n", + " style=[\"r--\", \"r--.\", \"b-\", \"b-*\"])\n", + "plt.legend(loc=\"lower left\") # not in the book\n", + "save_fig(\"keras_learning_curves_plot\") # not in the book\n", "plt.show()" ] }, @@ -717,7 +816,16 @@ "metadata": {}, "outputs": [], "source": [ - "model.evaluate(X_test, y_test)" + "# not in the book – shows how to shift the training curve by -1/2 epoch\n", + "plt.figure(figsize=(8, 5))\n", + "for key, style in zip(history.history, [\"r--\", \"r--.\", \"b-\", \"b-*\"]):\n", + " epochs = np.array(history.epoch) + (0 if key.startswith(\"val_\") else -0.5)\n", + " plt.plot(epochs, history.history[key], style, label=key)\n", + "plt.xlabel(\"Epoch\")\n", + "plt.axis([-0.5, 29, 0., 1])\n", + "plt.legend(loc=\"lower left\")\n", + "plt.grid()\n", + "plt.show()" ] }, { @@ -725,33 +833,41 @@ "execution_count": 43, "metadata": {}, "outputs": [], + "source": [ + "model.evaluate(X_test, y_test)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### Using the model to make predictions" + ] + }, + { + "cell_type": "code", + "execution_count": 44, + "metadata": {}, + "outputs": [], "source": [ "X_new = X_test[:3]\n", "y_proba = model.predict(X_new)\n", "y_proba.round(2)" ] }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "**Warning**: `model.predict_classes(X_new)` is deprecated. It is replaced with `np.argmax(model.predict(X_new), axis=-1)`." - ] - }, { "cell_type": "code", - "execution_count": 44, + "execution_count": 45, "metadata": {}, "outputs": [], "source": [ - "#y_pred = model.predict_classes(X_new) # deprecated\n", - "y_pred = np.argmax(model.predict(X_new), axis=-1)\n", + "y_pred = y_proba.argmax(axis=-1)\n", "y_pred" ] }, { "cell_type": "code", - "execution_count": 45, + "execution_count": 46, "metadata": {}, "outputs": [], "source": [ @@ -760,7 +876,7 @@ }, { "cell_type": "code", - "execution_count": 46, + "execution_count": 47, "metadata": {}, "outputs": [], "source": [ @@ -770,16 +886,17 @@ }, { "cell_type": "code", - "execution_count": 47, + "execution_count": 48, "metadata": {}, "outputs": [], "source": [ + "# not in the book – this cell generates and saves Figure 10–12\n", "plt.figure(figsize=(7.2, 2.4))\n", "for index, image in enumerate(X_new):\n", " plt.subplot(1, 3, index + 1)\n", " plt.imshow(image, cmap=\"binary\", interpolation=\"nearest\")\n", " plt.axis('off')\n", - " plt.title(class_names[y_test[index]], fontsize=12)\n", + " plt.title(class_names[y_test[index]])\n", "plt.subplots_adjust(wspace=0.2, hspace=0.5)\n", "save_fig('fashion_mnist_images_plot', tight_layout=False)\n", "plt.show()" @@ -789,7 +906,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# Regression MLP" + "## Building a Regression MLP Using the Sequential API" ] }, { @@ -799,35 +916,18 @@ "Let's load, split and scale the California housing dataset (the original one, not the modified one as in chapter 2):" ] }, - { - "cell_type": "code", - "execution_count": 48, - "metadata": {}, - "outputs": [], - "source": [ - "from sklearn.datasets import fetch_california_housing\n", - "from sklearn.model_selection import train_test_split\n", - "from sklearn.preprocessing import StandardScaler\n", - "\n", - "housing = fetch_california_housing()\n", - "\n", - "X_train_full, X_test, y_train_full, y_test = train_test_split(housing.data, housing.target, random_state=42)\n", - "X_train, X_valid, y_train, y_valid = train_test_split(X_train_full, y_train_full, random_state=42)\n", - "\n", - "scaler = StandardScaler()\n", - "X_train = scaler.fit_transform(X_train)\n", - "X_valid = scaler.transform(X_valid)\n", - "X_test = scaler.transform(X_test)" - ] - }, { "cell_type": "code", "execution_count": 49, "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "# not in the book – load and split the California housing dataset, like earlier\n", + "housing = fetch_california_housing()\n", + "X_train_full, X_test, y_train_full, y_test = train_test_split(\n", + " housing.data, housing.target, random_state=42)\n", + "X_train, X_valid, y_train, y_valid = train_test_split(\n", + " X_train_full, y_train_full, random_state=42)" ] }, { @@ -836,13 +936,21 @@ "metadata": {}, "outputs": [], "source": [ + "tf.random.set_seed(42)\n", + "norm_layer = tf.keras.layers.Normalization(input_shape=X_train.shape[1:])\n", "model = tf.keras.Sequential([\n", - " tf.keras.layers.Dense(30, activation=\"relu\", input_shape=X_train.shape[1:]),\n", + " norm_layer,\n", + " tf.keras.layers.Dense(50, activation=\"relu\"),\n", + " tf.keras.layers.Dense(50, activation=\"relu\"),\n", + " tf.keras.layers.Dense(50, activation=\"relu\"),\n", " tf.keras.layers.Dense(1)\n", "])\n", - "model.compile(loss=\"mean_squared_error\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", - "history = model.fit(X_train, y_train, epochs=20, validation_data=(X_valid, y_valid))\n", - "mse_test = model.evaluate(X_test, y_test)\n", + "optimizer = tf.keras.optimizers.Adam(learning_rate=1e-3)\n", + "model.compile(loss=\"mse\", optimizer=optimizer, metrics=[\"RootMeanSquaredError\"])\n", + "norm_layer.adapt(X_train)\n", + "history = model.fit(X_train, y_train, epochs=20,\n", + " validation_data=(X_valid, y_valid))\n", + "mse_test, rmse_test = model.evaluate(X_test, y_test)\n", "X_new = X_test[:3]\n", "y_pred = model.predict(X_new)" ] @@ -853,10 +961,7 @@ "metadata": {}, "outputs": [], "source": [ - "plt.plot(pd.DataFrame(history.history))\n", - "plt.grid(True)\n", - "plt.gca().set_ylim(0, 1)\n", - "plt.show()" + "rmse_test" ] }, { @@ -872,7 +977,7 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "# Functional API" + "## Building Complex Models Using the Functional API" ] }, { @@ -888,7 +993,8 @@ "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", + "# not in the book – reset the name counters and make the code reproducible\n", + "tf.keras.backend.clear_session()\n", "tf.random.set_seed(42)" ] }, @@ -898,11 +1004,19 @@ "metadata": {}, "outputs": [], "source": [ + "normalization_layer = tf.keras.layers.Normalization()\n", + "hidden_layer1 = tf.keras.layers.Dense(30, activation=\"relu\")\n", + "hidden_layer2 = tf.keras.layers.Dense(30, activation=\"relu\")\n", + "concat_layer = tf.keras.layers.Concatenate()\n", + "output_layer = tf.keras.layers.Dense(1)\n", + "\n", "input_ = tf.keras.layers.Input(shape=X_train.shape[1:])\n", - "hidden1 = tf.keras.layers.Dense(30, activation=\"relu\")(input_)\n", - "hidden2 = tf.keras.layers.Dense(30, activation=\"relu\")(hidden1)\n", - "concat = tf.keras.layers.concatenate([input_, hidden2])\n", - "output = tf.keras.layers.Dense(1)(concat)\n", + "normalized = normalization_layer(input_)\n", + "hidden1 = hidden_layer1(normalized)\n", + "hidden2 = hidden_layer2(hidden1)\n", + "concat = concat_layer([input_, hidden2])\n", + "output = output_layer(concat)\n", + "\n", "model = tf.keras.Model(inputs=[input_], outputs=[output])" ] }, @@ -921,7 +1035,9 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mean_squared_error\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", + "optimizer = tf.keras.optimizers.Adam(learning_rate=1e-3)\n", + "model.compile(loss=\"mse\", optimizer=optimizer, metrics=[\"RootMeanSquaredError\"])\n", + "normalization_layer.adapt(X_train)\n", "history = model.fit(X_train, y_train, epochs=20,\n", " validation_data=(X_valid, y_valid))\n", "mse_test = model.evaluate(X_test, y_test)\n", @@ -941,8 +1057,7 @@ "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "tf.random.set_seed(42) # not in the book" ] }, { @@ -951,13 +1066,17 @@ "metadata": {}, "outputs": [], "source": [ - "input_A = tf.keras.layers.Input(shape=[5], name=\"wide_input\")\n", - "input_B = tf.keras.layers.Input(shape=[6], name=\"deep_input\")\n", - "hidden1 = tf.keras.layers.Dense(30, activation=\"relu\")(input_B)\n", + "input_wide = tf.keras.layers.Input(shape=[5]) # features 0 to 4\n", + "input_deep = tf.keras.layers.Input(shape=[6]) # features 2 to 7\n", + "norm_layer_wide = tf.keras.layers.Normalization()\n", + "norm_layer_deep = tf.keras.layers.Normalization()\n", + "norm_wide = norm_layer_wide(input_wide)\n", + "norm_deep = norm_layer_deep(input_deep)\n", + "hidden1 = tf.keras.layers.Dense(30, activation=\"relu\")(norm_deep)\n", "hidden2 = tf.keras.layers.Dense(30, activation=\"relu\")(hidden1)\n", - "concat = tf.keras.layers.concatenate([input_A, hidden2])\n", - "output = tf.keras.layers.Dense(1, name=\"output\")(concat)\n", - "model = tf.keras.Model(inputs=[input_A, input_B], outputs=[output])" + "concat = tf.keras.layers.concatenate([norm_wide, hidden2])\n", + "output = tf.keras.layers.Dense(1)(concat)\n", + "model = tf.keras.Model(inputs=[input_wide, input_deep], outputs=[output])" ] }, { @@ -966,17 +1085,20 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", + "optimizer = tf.keras.optimizers.Adam(learning_rate=1e-3)\n", + "model.compile(loss=\"mse\", optimizer=optimizer, metrics=[\"RootMeanSquaredError\"])\n", "\n", - "X_train_A, X_train_B = X_train[:, :5], X_train[:, 2:]\n", - "X_valid_A, X_valid_B = X_valid[:, :5], X_valid[:, 2:]\n", - "X_test_A, X_test_B = X_test[:, :5], X_test[:, 2:]\n", - "X_new_A, X_new_B = X_test_A[:3], X_test_B[:3]\n", + "X_train_wide, X_train_deep = X_train[:, :5], X_train[:, 2:]\n", + "X_valid_wide, X_valid_deep = X_valid[:, :5], X_valid[:, 2:]\n", + "X_test_wide, X_test_deep = X_test[:, :5], X_test[:, 2:]\n", + "X_new_wide, X_new_deep = X_test_wide[:3], X_test_deep[:3]\n", "\n", - "history = model.fit((X_train_A, X_train_B), y_train, epochs=20,\n", - " validation_data=((X_valid_A, X_valid_B), y_valid))\n", - "mse_test = model.evaluate((X_test_A, X_test_B), y_test)\n", - "y_pred = model.predict((X_new_A, X_new_B))" + "norm_layer_wide.adapt(X_train_wide)\n", + "norm_layer_deep.adapt(X_train_deep)\n", + "history = model.fit((X_train_wide, X_train_deep), y_train, epochs=20,\n", + " validation_data=((X_valid_wide, X_valid_deep), y_valid))\n", + "mse_test = model.evaluate((X_test_wide, X_test_deep), y_test)\n", + "y_pred = model.predict((X_new_wide, X_new_deep))" ] }, { @@ -992,7 +1114,7 @@ "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", + "tf.keras.backend.clear_session()\n", "tf.random.set_seed(42)" ] }, @@ -1002,15 +1124,19 @@ "metadata": {}, "outputs": [], "source": [ - "input_A = tf.keras.layers.Input(shape=[5], name=\"wide_input\")\n", - "input_B = tf.keras.layers.Input(shape=[6], name=\"deep_input\")\n", - "hidden1 = tf.keras.layers.Dense(30, activation=\"relu\")(input_B)\n", + "input_wide = tf.keras.layers.Input(shape=[5]) # features 0 to 4\n", + "input_deep = tf.keras.layers.Input(shape=[6]) # features 2 to 7\n", + "norm_layer_wide = tf.keras.layers.Normalization()\n", + "norm_layer_deep = tf.keras.layers.Normalization()\n", + "norm_wide = norm_layer_wide(input_wide)\n", + "norm_deep = norm_layer_deep(input_deep)\n", + "hidden1 = tf.keras.layers.Dense(30, activation=\"relu\")(norm_deep)\n", "hidden2 = tf.keras.layers.Dense(30, activation=\"relu\")(hidden1)\n", - "concat = tf.keras.layers.concatenate([input_A, hidden2])\n", - "output = tf.keras.layers.Dense(1, name=\"main_output\")(concat)\n", - "aux_output = tf.keras.layers.Dense(1, name=\"aux_output\")(hidden2)\n", - "model = tf.keras.Model(inputs=[input_A, input_B],\n", - " outputs=[output, aux_output])" + "concat = tf.keras.layers.concatenate([norm_wide, hidden2])\n", + "output = tf.keras.layers.Dense(1)(concat)\n", + "aux_output = tf.keras.layers.Dense(1)(hidden2)\n", + "model = tf.keras.Model(inputs=[input_wide, input_deep],\n", + " outputs=[output, aux_output])" ] }, { @@ -1019,7 +1145,9 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=[\"mse\", \"mse\"], loss_weights=[0.9, 0.1], optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))" + "optimizer = tf.keras.optimizers.Adam(learning_rate=1e-3)\n", + "model.compile(loss=(\"mse\", \"mse\"), loss_weights=(0.9, 0.1), optimizer=optimizer,\n", + " metrics=[\"RootMeanSquaredError\"])" ] }, { @@ -1028,8 +1156,12 @@ "metadata": {}, "outputs": [], "source": [ - "history = model.fit([X_train_A, X_train_B], [y_train, y_train], epochs=20,\n", - " validation_data=([X_valid_A, X_valid_B], [y_valid, y_valid]))" + "norm_layer_wide.adapt(X_train_wide)\n", + "norm_layer_deep.adapt(X_train_deep)\n", + "history = model.fit(\n", + " (X_train_wide, X_train_deep), (y_train, y_train), epochs=20,\n", + " validation_data=((X_valid_wide, X_valid_deep), (y_valid, y_valid))\n", + ")" ] }, { @@ -1038,16 +1170,8 @@ "metadata": {}, "outputs": [], "source": [ - "total_loss, main_loss, aux_loss = model.evaluate(\n", - " [X_test_A, X_test_B], [y_test, y_test])\n", - "y_pred_main, y_pred_aux = model.predict([X_new_A, X_new_B])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# The subclassing API" + "eval_results = model.evaluate((X_test_wide, X_test_deep), (y_test, y_test))\n", + "weighted_sum_of_losses, main_loss, aux_loss, main_rmse, aux_rmse = eval_results" ] }, { @@ -1056,24 +1180,7 @@ "metadata": {}, "outputs": [], "source": [ - "class WideAndDeepModel(tf.keras.Model):\n", - " def __init__(self, units=30, activation=\"relu\", **kwargs):\n", - " super().__init__(**kwargs)\n", - " self.hidden1 = tf.keras.layers.Dense(units, activation=activation)\n", - " self.hidden2 = tf.keras.layers.Dense(units, activation=activation)\n", - " self.main_output = tf.keras.layers.Dense(1)\n", - " self.aux_output = tf.keras.layers.Dense(1)\n", - " \n", - " def call(self, inputs):\n", - " input_A, input_B = inputs\n", - " hidden1 = self.hidden1(input_B)\n", - " hidden2 = self.hidden2(hidden1)\n", - " concat = tf.keras.layers.concatenate([input_A, hidden2])\n", - " main_output = self.main_output(concat)\n", - " aux_output = self.aux_output(hidden2)\n", - " return main_output, aux_output\n", - "\n", - "model = WideAndDeepModel(30, activation=\"relu\")" + "y_pred_main, y_pred_aux = model.predict((X_new_wide, X_new_deep))" ] }, { @@ -1082,18 +1189,46 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mse\", loss_weights=[0.9, 0.1], optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", - "history = model.fit((X_train_A, X_train_B), (y_train, y_train), epochs=10,\n", - " validation_data=((X_valid_A, X_valid_B), (y_valid, y_valid)))\n", - "total_loss, main_loss, aux_loss = model.evaluate((X_test_A, X_test_B), (y_test, y_test))\n", - "y_pred_main, y_pred_aux = model.predict((X_new_A, X_new_B))" + "y_pred_tuple = model.predict((X_new_wide, X_new_deep))\n", + "y_pred = dict(zip(model.output_names, y_pred_tuple))" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "# Saving and Restoring" + "## Using the Subclassing API to Build Dynamic Models" + ] + }, + { + "cell_type": "code", + "execution_count": 67, + "metadata": {}, + "outputs": [], + "source": [ + "class WideAndDeepModel(tf.keras.Model):\n", + " def __init__(self, units=30, activation=\"relu\", **kwargs):\n", + " super().__init__(**kwargs) # needed to support naming the model\n", + " self.norm_layer_wide = tf.keras.layers.Normalization()\n", + " self.norm_layer_deep = tf.keras.layers.Normalization()\n", + " self.hidden1 = tf.keras.layers.Dense(units, activation=activation)\n", + " self.hidden2 = tf.keras.layers.Dense(units, activation=activation)\n", + " self.main_output = tf.keras.layers.Dense(1)\n", + " self.aux_output = tf.keras.layers.Dense(1)\n", + " \n", + " def call(self, inputs):\n", + " input_wide, input_deep = inputs\n", + " norm_wide = self.norm_layer_wide(input_wide)\n", + " norm_deep = self.norm_layer_deep(input_deep)\n", + " hidden1 = self.hidden1(norm_deep)\n", + " hidden2 = self.hidden2(hidden1)\n", + " concat = tf.keras.layers.concatenate([norm_wide, hidden2])\n", + " output = self.main_output(concat)\n", + " aux_output = self.aux_output(hidden2)\n", + " return output, aux_output\n", + "\n", + "tf.random.set_seed(42) # not in the book – just for reproducibility\n", + "model = WideAndDeepModel(30, activation=\"relu\", name=\"my_cool_model\")" ] }, { @@ -1102,8 +1237,24 @@ "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "optimizer = tf.keras.optimizers.Adam(learning_rate=1e-3)\n", + "model.compile(loss=\"mse\", loss_weights=[0.9, 0.1], optimizer=optimizer,\n", + " metrics=[\"RootMeanSquaredError\"])\n", + "model.norm_layer_wide.adapt(X_train_wide)\n", + "model.norm_layer_deep.adapt(X_train_deep)\n", + "history = model.fit(\n", + " (X_train_wide, X_train_deep), (y_train, y_train), epochs=10,\n", + " validation_data=((X_valid_wide, X_valid_deep), (y_valid, y_valid)))\n", + "eval_results = model.evaluate((X_test_wide, X_test_deep), (y_test, y_test))\n", + "weighted_sum_of_losses, main_loss, aux_loss, main_rmse, aux_rmse = eval_results\n", + "y_pred_main, y_pred_aux = model.predict((X_new_wide, X_new_deep))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Saving and Restoring a Model" ] }, { @@ -1112,11 +1263,11 @@ "metadata": {}, "outputs": [], "source": [ - "model = tf.keras.Sequential([\n", - " tf.keras.layers.Dense(30, activation=\"relu\", input_shape=[8]),\n", - " tf.keras.layers.Dense(30, activation=\"relu\"),\n", - " tf.keras.layers.Dense(1)\n", - "]) " + "# not in the book – delete the directory, in case it already exists\n", + "\n", + "import shutil\n", + "\n", + "shutil.rmtree(\"my_keras_model\", ignore_errors=True)" ] }, { @@ -1125,9 +1276,7 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", - "history = model.fit(X_train, y_train, epochs=10, validation_data=(X_valid, y_valid))\n", - "mse_test = model.evaluate(X_test, y_test)" + "model.save(\"my_keras_model\")" ] }, { @@ -1136,7 +1285,9 @@ "metadata": {}, "outputs": [], "source": [ - "model.save(\"my_keras_model.h5\")" + "# not in the book – show the contents of the my_keras_model/ directory\n", + "for path in sorted(Path(\"my_keras_model\").glob(\"**/*\")):\n", + " print(path)" ] }, { @@ -1145,7 +1296,8 @@ "metadata": {}, "outputs": [], "source": [ - "model = tf.keras.models.load_model(\"my_keras_model.h5\")" + "model = tf.keras.models.load_model(\"my_keras_model\")\n", + "y_pred_main, y_pred_aux = model.predict((X_new_wide, X_new_deep))" ] }, { @@ -1154,7 +1306,7 @@ "metadata": {}, "outputs": [], "source": [ - "model.predict(X_new)" + "model.save_weights(\"my_weights\")" ] }, { @@ -1163,7 +1315,7 @@ "metadata": {}, "outputs": [], "source": [ - "model.save_weights(\"my_keras_weights.ckpt\")" + "model.load_weights(\"my_weights\")" ] }, { @@ -1172,14 +1324,16 @@ "metadata": {}, "outputs": [], "source": [ - "model.load_weights(\"my_keras_weights.ckpt\")" + "# not in the book – show the list of my_weights.* files\n", + "for path in sorted(Path().glob(\"my_weights.*\")):\n", + " print(path)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "# Using Callbacks during Training" + "## Using Callbacks" ] }, { @@ -1188,9 +1342,7 @@ "metadata": {}, "outputs": [], "source": [ - "tf.keras.backend.clear_session()\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "shutil.rmtree(\"my_checkpoints\", ignore_errors=True) # not in the book" ] }, { @@ -1199,11 +1351,12 @@ "metadata": {}, "outputs": [], "source": [ - "model = tf.keras.Sequential([\n", - " tf.keras.layers.Dense(30, activation=\"relu\", input_shape=[8]),\n", - " tf.keras.layers.Dense(30, activation=\"relu\"),\n", - " tf.keras.layers.Dense(1)\n", - "]) " + "checkpoint_cb = tf.keras.callbacks.ModelCheckpoint(\"my_checkpoints\",\n", + " save_weights_only=True)\n", + "history = model.fit(\n", + " (X_train_wide, X_train_deep), (y_train, y_train), epochs=10,\n", + " validation_data=((X_valid_wide, X_valid_deep), (y_valid, y_valid)),\n", + " callbacks=[checkpoint_cb])" ] }, { @@ -1212,13 +1365,12 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", - "checkpoint_cb = tf.keras.callbacks.ModelCheckpoint(\"my_keras_model.h5\", save_best_only=True)\n", - "history = model.fit(X_train, y_train, epochs=10,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[checkpoint_cb])\n", - "model = tf.keras.models.load_model(\"my_keras_model.h5\") # rollback to best model\n", - "mse_test = model.evaluate(X_test, y_test)" + "early_stopping_cb = tf.keras.callbacks.EarlyStopping(patience=10,\n", + " restore_best_weights=True)\n", + "history = model.fit(\n", + " (X_train_wide, X_train_deep), (y_train, y_train), epochs=100,\n", + " validation_data=((X_valid_wide, X_valid_deep), (y_valid, y_valid)),\n", + " callbacks=[checkpoint_cb, early_stopping_cb])" ] }, { @@ -1227,13 +1379,10 @@ "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))\n", - "early_stopping_cb = tf.keras.callbacks.EarlyStopping(patience=10,\n", - " restore_best_weights=True)\n", - "history = model.fit(X_train, y_train, epochs=100,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[checkpoint_cb, early_stopping_cb])\n", - "mse_test = model.evaluate(X_test, y_test)" + "class PrintValTrainRatioCallback(tf.keras.callbacks.Callback):\n", + " def on_epoch_end(self, epoch, logs):\n", + " ratio = logs[\"val_loss\"] / logs[\"loss\"]\n", + " print(f\"Epoch={epoch}, val/train={ratio:.2f}\")" ] }, { @@ -1242,9 +1391,25 @@ "metadata": {}, "outputs": [], "source": [ - "class PrintValTrainRatioCallback(tf.keras.callbacks.Callback):\n", - " def on_epoch_end(self, epoch, logs):\n", - " print(\"\\nval/train: {:.2f}\".format(logs[\"val_loss\"] / logs[\"loss\"]))" + "val_train_ratio_cb = PrintValTrainRatioCallback()\n", + "history = model.fit(\n", + " (X_train_wide, X_train_deep), (y_train, y_train), epochs=10,\n", + " validation_data=((X_valid_wide, X_valid_deep), (y_valid, y_valid)),\n", + " callbacks=[val_train_ratio_cb], verbose=0)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Using TensorBoard for Visualization" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "TensorBoard is preinstalled on Colab, but not the `tensorboard-plugin-profile`, so let's install it:" ] }, { @@ -1253,41 +1418,34 @@ "metadata": {}, "outputs": [], "source": [ - "val_train_ratio_cb = PrintValTrainRatioCallback()\n", - "history = model.fit(X_train, y_train, epochs=1,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[val_train_ratio_cb])" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "# TensorBoard" + "if \"google.colab\" in sys.modules: # not in the book\n", + " %pip install -q -U tensorboard-plugin-profile" ] }, { "cell_type": "code", - "execution_count": 3, - "metadata": {}, + "execution_count": 82, + "metadata": { + "tags": [] + }, "outputs": [], "source": [ - "root_logdir = Path() / \"my_logs\"" + "shutil.rmtree(\"my_logs\", ignore_errors=True)" ] }, { "cell_type": "code", - "execution_count": 4, + "execution_count": 83, "metadata": {}, "outputs": [], "source": [ - "def get_run_logdir():\n", - " import time\n", - " run_id = time.strftime(\"run_%Y_%m_%d-%H_%M_%S\")\n", - " return root_logdir / run_id\n", + "from pathlib import Path\n", + "from time import strftime\n", "\n", - "run_logdir = get_run_logdir()\n", - "run_logdir" + "def get_run_logdir(root_logdir=\"my_logs\"):\n", + " return Path(root_logdir) / strftime(\"run_%Y_%m_%d_%H_%M_%S\")\n", + "\n", + "run_logdir = get_run_logdir()" ] }, { @@ -1296,9 +1454,19 @@ "metadata": {}, "outputs": [], "source": [ + "# not in the book – builds the first regression model we used earlier\n", "tf.keras.backend.clear_session()\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "tf.random.set_seed(42)\n", + "norm_layer = tf.keras.layers.Normalization(input_shape=X_train.shape[1:])\n", + "model = tf.keras.Sequential([\n", + " norm_layer,\n", + " tf.keras.layers.Dense(30, activation=\"relu\"),\n", + " tf.keras.layers.Dense(30, activation=\"relu\"),\n", + " tf.keras.layers.Dense(1)\n", + "])\n", + "optimizer = tf.keras.optimizers.SGD(learning_rate=1e-3)\n", + "model.compile(loss=\"mse\", optimizer=optimizer, metrics=[\"RootMeanSquaredError\"])\n", + "norm_layer.adapt(X_train)" ] }, { @@ -1307,12 +1475,11 @@ "metadata": {}, "outputs": [], "source": [ - "model = tf.keras.Sequential([\n", - " tf.keras.layers.Dense(30, activation=\"relu\", input_shape=[8]),\n", - " tf.keras.layers.Dense(30, activation=\"relu\"),\n", - " tf.keras.layers.Dense(1)\n", - "]) \n", - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3))" + "tensorboard_cb = tf.keras.callbacks.TensorBoard(run_logdir,\n", + " profile_batch=(100, 200))\n", + "history = model.fit(X_train, y_train, epochs=20,\n", + " validation_data=(X_valid, y_valid),\n", + " callbacks=[tensorboard_cb])" ] }, { @@ -1321,25 +1488,16 @@ "metadata": {}, "outputs": [], "source": [ - "tensorboard_cb = tf.keras.callbacks.TensorBoard(run_logdir)\n", - "history = model.fit(X_train, y_train, epochs=30,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[checkpoint_cb, tensorboard_cb])" + "print(\"my_logs\")\n", + "for path in sorted(Path(\"my_logs\").glob(\"**/*\")):\n", + " print(\" \" * (len(path.parts) - 1) + path.parts[-1])" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "To start the TensorBoard server, one option is to open a terminal, if needed activate the virtualenv where you installed TensorBoard, go to this notebook's directory, then type:\n", - "\n", - "```bash\n", - "$ tensorboard --logdir=./my_logs --port=6006\n", - "```\n", - "\n", - "You can then open your web browser to [localhost:6006](http://localhost:6006) and use TensorBoard. Once you are done, press Ctrl-C in the terminal window, this will shutdown the TensorBoard server.\n", - "\n", - "Alternatively, you can load TensorBoard's Jupyter extension and run it like this:" + "Let's load the `tensorboard` Jupyter extension and start the TensorBoard server: " ] }, { @@ -1349,17 +1507,39 @@ "outputs": [], "source": [ "%load_ext tensorboard\n", - "%tensorboard --logdir=./my_logs --port=6006" + "%tensorboard --logdir=./my_logs" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "**Note**: if you prefer to access TensorBoard in a separate tab, click the \"localhost:6006\" link below:" ] }, { "cell_type": "code", - "execution_count": 5, + "execution_count": 88, "metadata": {}, "outputs": [], "source": [ - "run_logdir2 = get_run_logdir()\n", - "run_logdir2" + "# not in the book\n", + "\n", + "if \"google.colab\" in sys.modules:\n", + " from google.colab import output\n", + "\n", + " output.serve_kernel_port_as_window(6006)\n", + "else:\n", + " from IPython.core.display import display, HTML\n", + "\n", + " display(HTML('http://localhost:6006/'))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You can use also visualize histograms, images, text, and even listen to audio using TensorBoard:" ] }, { @@ -1368,9 +1548,31 @@ "metadata": {}, "outputs": [], "source": [ - "tf.keras.backend.clear_session()\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "test_logdir = get_run_logdir()\n", + "writer = tf.summary.create_file_writer(str(test_logdir))\n", + "with writer.as_default():\n", + " for step in range(1, 1000 + 1):\n", + " tf.summary.scalar(\"my_scalar\", np.sin(step / 10), step=step)\n", + " \n", + " data = (np.random.randn(100) + 2) * step / 100 # gets larger\n", + " tf.summary.histogram(\"my_hist\", data, buckets=50, step=step)\n", + " \n", + " images = np.random.rand(2, 32, 32, 3) * step / 1000 # gets brighter\n", + " tf.summary.image(\"my_images\", images, step=step)\n", + " \n", + " texts = [\"The step is \" + str(step), \"Its square is \" + str(step ** 2)]\n", + " tf.summary.text(\"my_text\", texts, step=step)\n", + " \n", + " sine_wave = tf.math.sin(tf.range(12000) / 48000 * 2 * np.pi * step)\n", + " audio = tf.reshape(tf.cast(sine_wave, tf.float32), [1, -1, 1])\n", + " tf.summary.audio(\"my_audio\", audio, sample_rate=48000, step=step)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You can share your TensorBoard logs with the world by uploading them to https://tensorboard.dev/. For this, you can run the `tensorboard dev upload` command, with the `--logdir` and `--one_shot` options, and optionally the `--name` and `--description` options. The first time, it will ask you to accept Google's Terms of Service, and to authenticate. This requires user input. Colab supports user input from shell commands, but the main other Jupyter environments do not, so for them we use a hackish workaround (alternatively, you could run the command in a terminal window, after you make sure to activate this project's conda environment and move to this notebook's directory)." ] }, { @@ -1379,12 +1581,29 @@ "metadata": {}, "outputs": [], "source": [ - "model = tf.keras.Sequential([\n", - " tf.keras.layers.Dense(30, activation=\"relu\", input_shape=[8]),\n", - " tf.keras.layers.Dense(30, activation=\"relu\"),\n", - " tf.keras.layers.Dense(1)\n", - "]) \n", - "model.compile(loss=\"mse\", optimizer=tf.keras.optimizers.SGD(learning_rate=0.05))" + "# not in the book\n", + "\n", + "if \"google.colab\" in sys.modules:\n", + " !tensorboard dev upload --logdir ./my_logs --one_shot \\\n", + " --name \"Quick test\" --description \"This is a test\" \n", + "else:\n", + " from tensorboard.main import run_main\n", + "\n", + " argv = \"tensorboard dev upload --logdir ./my_logs --one_shot\".split()\n", + " argv += [\"--name\", \"Quick test\", \"--description\", \"This is a test\"]\n", + " try:\n", + " original_sys_argv_and_sys_exit = sys.argv, sys.exit\n", + " sys.argv, sys.exit = argv, lambda status: None\n", + " run_main()\n", + " finally:\n", + " sys.argv, sys.exit = original_sys_argv_and_sys_exit" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "You can get list your published experiments:" ] }, { @@ -1393,24 +1612,25 @@ "metadata": {}, "outputs": [], "source": [ - "tensorboard_cb = tf.keras.callbacks.TensorBoard(run_logdir2)\n", - "history = model.fit(X_train, y_train, epochs=30,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[checkpoint_cb, tensorboard_cb])" + "!tensorboard dev list" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "Notice how TensorBoard now sees two runs, and you can compare the learning curves." + "To delete an experiment, use the following command:\n", + "\n", + "```python\n", + "!tensorboard dev delete --experiment_id \n", + "```" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "Check out the other available logging options:" + "When you stop this Jupyter kernel (a.k.a. Runtime), it will automatically stop the TensorBoard server as well. Another way to stop the TensorBoard server is to kill it, if you are running on Linux or MacOSX. First, you need to find its process ID:" ] }, { @@ -1419,14 +1639,38 @@ "metadata": {}, "outputs": [], "source": [ - "help(tf.keras.callbacks.TensorBoard.__init__)" + "# not in the book – extra material\n", + "\n", + "from tensorboard import notebook\n", + "\n", + "notebook.list()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ - "# Hyperparameter Tuning" + "Next you can use the following command on Linux or MacOSX, replacing `` with the pid listed above:\n", + "\n", + " !kill \n", + "\n", + "On Windows:\n", + "\n", + " !taskkill /F /PID " + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Fine-Tuning Neural Network Hyperparameters" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In this section we'll use the Fashion MNIST dataset again:" ] }, { @@ -1435,9 +1679,9 @@ "metadata": {}, "outputs": [], "source": [ - "tf.keras.backend.clear_session()\n", - "np.random.seed(42)\n", - "tf.random.set_seed(42)" + "(X_train_full, y_train_full), (X_test, y_test) = fashion_mnist\n", + "X_train, y_train = X_train_full[:-5000], y_train_full[:-5000]\n", + "X_valid, y_valid = X_train_full[-5000:], y_train_full[-5000:]" ] }, { @@ -1446,15 +1690,8 @@ "metadata": {}, "outputs": [], "source": [ - "def build_model(n_hidden=1, n_neurons=30, learning_rate=3e-3, input_shape=[8]):\n", - " model = tf.keras.Sequential()\n", - " model.add(tf.keras.layers.InputLayer(input_shape=input_shape))\n", - " for layer in range(n_hidden):\n", - " model.add(tf.keras.layers.Dense(n_neurons, activation=\"relu\"))\n", - " model.add(tf.keras.layers.Dense(1))\n", - " optimizer = tf.keras.optimizers.SGD(learning_rate=learning_rate)\n", - " model.compile(loss=\"mse\", optimizer=optimizer)\n", - " return model" + "tf.keras.backend.clear_session()\n", + "tf.random.set_seed(42)" ] }, { @@ -1463,7 +1700,8 @@ "metadata": {}, "outputs": [], "source": [ - "keras_reg = tf.keras.wrappers.scikit_learn.KerasRegressor(build_model)" + "if \"google.colab\" in sys.modules:\n", + " %pip install -q -U keras_tuner" ] }, { @@ -1472,9 +1710,27 @@ "metadata": {}, "outputs": [], "source": [ - "keras_reg.fit(X_train, y_train, epochs=100,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[tf.keras.callbacks.EarlyStopping(patience=10)])" + "import keras_tuner as kt\n", + "\n", + "def build_model(hp):\n", + " n_hidden = hp.Int(\"n_hidden\", min_value=0, max_value=8, default=2)\n", + " n_neurons = hp.Int(\"n_neurons\", min_value=16, max_value=256)\n", + " learning_rate = hp.Float(\"learning_rate\", min_value=1e-4, max_value=1e-2,\n", + " sampling=\"log\")\n", + " optimizer = hp.Choice(\"optimizer\", values=[\"sgd\", \"adam\"])\n", + " if optimizer == \"sgd\":\n", + " optimizer = tf.keras.optimizers.SGD(learning_rate=learning_rate)\n", + " else:\n", + " optimizer = tf.keras.optimizers.Adam(learning_rate=learning_rate)\n", + "\n", + " model = tf.keras.Sequential()\n", + " model.add(tf.keras.layers.Flatten())\n", + " for _ in range(n_hidden):\n", + " model.add(tf.keras.layers.Dense(n_neurons, activation=\"relu\"))\n", + " model.add(tf.keras.layers.Dense(10, activation=\"softmax\"))\n", + " model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer,\n", + " metrics=[\"accuracy\"])\n", + " return model" ] }, { @@ -1483,7 +1739,11 @@ "metadata": {}, "outputs": [], "source": [ - "mse_test = keras_reg.score(X_test, y_test)" + "random_search_tuner = kt.RandomSearch(\n", + " build_model, objective=\"val_accuracy\", max_trials=5, overwrite=True,\n", + " directory=\"my_fashion_mnist\", project_name=\"my_rnd_search\", seed=42)\n", + "random_search_tuner.search(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid))" ] }, { @@ -1492,7 +1752,8 @@ "metadata": {}, "outputs": [], "source": [ - "y_pred = keras_reg.predict(X_new)" + "top3_models = random_search_tuner.get_best_models(num_models=3)\n", + "best_model = top3_models[0]" ] }, { @@ -1501,15 +1762,8 @@ "metadata": {}, "outputs": [], "source": [ - "np.random.seed(42)\n", - "tf.random.set_seed(42)" - ] - }, - { - "cell_type": "markdown", - "metadata": {}, - "source": [ - "**Warning**: the following cell crashes at the end of training. This seems to be caused by [Keras issue #13586](https://github.com/keras-team/keras/issues/13586), which was triggered by a recent change in Scikit-Learn. [Pull Request #13598](https://github.com/keras-team/keras/pull/13598) seems to fix the issue, so this problem should be resolved soon. In the meantime, I've added `.tolist()` and `.rvs(1000).tolist()` as workarounds." + "top3_params = random_search_tuner.get_best_hyperparameters(num_trials=3)\n", + "top3_params[0].values # best hyperparameter values" ] }, { @@ -1518,19 +1772,8 @@ "metadata": {}, "outputs": [], "source": [ - "from scipy.stats import reciprocal\n", - "from sklearn.model_selection import RandomizedSearchCV\n", - "\n", - "param_distribs = {\n", - " \"n_hidden\": [0, 1, 2, 3],\n", - " \"n_neurons\": np.arange(1, 100) .tolist(),\n", - " \"learning_rate\": reciprocal(3e-4, 3e-2) .rvs(1000).tolist(),\n", - "}\n", - "\n", - "rnd_search_cv = RandomizedSearchCV(keras_reg, param_distribs, n_iter=10, cv=3, verbose=2)\n", - "rnd_search_cv.fit(X_train, y_train, epochs=100,\n", - " validation_data=(X_valid, y_valid),\n", - " callbacks=[tf.keras.callbacks.EarlyStopping(patience=10)])" + "best_trial = random_search_tuner.oracle.get_best_trials(num_trials=1)[0]\n", + "best_trial.summary()" ] }, { @@ -1539,7 +1782,7 @@ "metadata": {}, "outputs": [], "source": [ - "rnd_search_cv.best_params_" + "best_trial.metrics.get_last_value(\"val_accuracy\")" ] }, { @@ -1548,7 +1791,8 @@ "metadata": {}, "outputs": [], "source": [ - "rnd_search_cv.best_score_" + "best_model.fit(X_train_full, y_train_full, epochs=10)\n", + "test_loss, test_accuracy = best_model.evaluate(X_test, y_test)" ] }, { @@ -1557,7 +1801,15 @@ "metadata": {}, "outputs": [], "source": [ - "rnd_search_cv.best_estimator_" + "class MyClassificationHyperModel(kt.HyperModel):\n", + " def build(self, hp):\n", + " return build_model(hp)\n", + "\n", + " def fit(self, hp, model, X, y, **kwargs):\n", + " if hp.Boolean(\"normalize\"):\n", + " norm_layer = tf.keras.layers.Normalization()\n", + " X = norm_layer(X)\n", + " return model.fit(X, y, **kwargs)" ] }, { @@ -1566,7 +1818,10 @@ "metadata": {}, "outputs": [], "source": [ - "rnd_search_cv.score(X_test, y_test)" + "hyperband_tuner = kt.Hyperband(\n", + " MyClassificationHyperModel(), objective=\"val_accuracy\", seed=42,\n", + " max_epochs=10, factor=3, hyperband_iterations=2,\n", + " overwrite=True, directory=\"my_fashion_mnist\", project_name=\"hyperband\")" ] }, { @@ -1575,19 +1830,36 @@ "metadata": {}, "outputs": [], "source": [ - "model = rnd_search_cv.best_estimator_.model\n", - "model" + "root_logdir = Path(hyperband_tuner.project_dir) / \"tensorboard\"\n", + "tensorboard_cb = tf.keras.callbacks.TensorBoard(root_logdir)\n", + "early_stopping_cb = tf.keras.callbacks.EarlyStopping(patience=2)\n", + "hyperband_tuner.search(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid),\n", + " callbacks=[early_stopping_cb, tensorboard_cb])" ] }, { "cell_type": "code", "execution_count": 106, - "metadata": { - "scrolled": true - }, + "metadata": {}, "outputs": [], "source": [ - "model.evaluate(X_test, y_test)" + "bayesian_opt_tuner = kt.BayesianOptimization(\n", + " MyClassificationHyperModel(), objective=\"val_accuracy\", seed=42,\n", + " max_trials=10, alpha=1e-4, beta=2.6,\n", + " overwrite=True, directory=\"my_fashion_mnist\", project_name=\"bayesian_opt\")\n", + "bayesian_opt_tuner.search(X_train, y_train, epochs=10,\n", + " validation_data=(X_valid, y_valid),\n", + " callbacks=[early_stopping_cb])" + ] + }, + { + "cell_type": "code", + "execution_count": 107, + "metadata": {}, + "outputs": [], + "source": [ + "%tensorboard --logdir {root_logdir}" ] }, { @@ -1608,7 +1880,20 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "See appendix A." + "1. Visit the [TensorFlow Playground](https://playground.tensorflow.org/) and play around with it, as described in this exercise.\n", + "2. Here is a neural network based on the original artificial neurons that computes _A_ ⊕ _B_ (where ⊕ represents the exclusive OR), using the fact that _A_ ⊕ _B_ = (_A_ ∧ ¬ _B_) ∨ (¬ _A_ ∧ _B_). There are other solutions—for example, using the fact that _A_ ⊕ _B_ = (_A_ ∨ _B_) ∧ ¬(_A_ ∧ _B_), or the fact that _A_ ⊕ _B_ = (_A_ ∨ _B_) ∧ (¬ _A_ ∨ ¬ _B_), and so on.
\n", + "3. A classical Perceptron will converge only if the dataset is linearly separable, and it won't be able to estimate class probabilities. In contrast, a Logistic Regression classifier will generally converge to a reasonably good solution even if the dataset is not linearly separable, and it will output class probabilities. If you change the Perceptron's activation function to the sigmoid activation function (or the softmax activation function if there are multiple neurons), and if you train it using Gradient Descent (or some other optimization algorithm minimizing the cost function, typically cross entropy), then it becomes equivalent to a Logistic Regression classifier.\n", + "4. The sigmoid activation function was a key ingredient in training the first MLPs because its derivative is always nonzero, so Gradient Descent can always roll down the slope. When the activation function is a step function, Gradient Descent cannot move, as there is no slope at all.\n", + "5. Popular activation functions include the step function, the sigmoid function, the hyperbolic tangent (tanh) function, and the Rectified Linear Unit (ReLU) function (see Figure 10-8). See Chapter 11 for other examples, such as ELU and variants of the ReLU function.\n", + "6. Considering the MLP described in the question, composed of one input layer with 10 passthrough neurons, followed by one hidden layer with 50 artificial neurons, and finally one output layer with 3 artificial neurons, where all artificial neurons use the ReLU activation function:\n", + " * The shape of the input matrix **X** is _m_ × 10, where _m_ represents the training batch size.\n", + " * The shape of the hidden layer's weight matrix **W**_h_ is 10 × 50, and the length of its bias vector **b**_h_ is 50.\n", + " * The shape of the output layer's weight matrix **W**_o_ is 50 × 3, and the length of its bias vector **b**_o_ is 3.\n", + " * The shape of the network's output matrix **Y** is _m_ × 3.\n", + " * **Y** = ReLU(ReLU(**X** **W**_h_ + **b**_h_) **W**_o_ + **b**_o_). Recall that the ReLU function just sets every negative number in the matrix to zero. Also note that when you are adding a bias vector to a matrix, it is added to every single row in the matrix, which is called _broadcasting_.\n", + "7. To classify email into spam or ham, you just need one neuron in the output layer of a neural network—for example, indicating the probability that the email is spam. You would typically use the sigmoid activation function in the output layer when estimating a probability. If instead you want to tackle MNIST, you need 10 neurons in the output layer, and you must replace the sigmoid function with the softmax activation function, which can handle multiple classes, outputting one probability per class. If you want your neural network to predict housing prices like in Chapter 2, then you need one output neuron, using no activation function at all in the output layer. Note: when the values to predict can vary by many orders of magnitude, you may want to predict the logarithm of the target value rather than the target value directly. Simply computing the exponential of the neural network's output will give you the estimated value (since exp(log _v_) = _v_).\n", + "8. Backpropagation is a technique used to train artificial neural networks. It first computes the gradients of the cost function with regard to every model parameter (all the weights and biases), then it performs a Gradient Descent step using these gradients. This backpropagation step is typically performed thousands or millions of times, using many training batches, until the model parameters converge to values that (hopefully) minimize the cost function. To compute the gradients, backpropagation uses reverse-mode autodiff (although it wasn't called that when backpropagation was invented, and it has been reinvented several times). Reverse-mode autodiff performs a forward pass through a computation graph, computing every node's value for the current training batch, and then it performs a reverse pass, computing all the gradients at once (see Appendix B for more details). So what's the difference? Well, backpropagation refers to the whole process of training an artificial neural network using multiple backpropagation steps, each of which computes gradients and uses them to perform a Gradient Descent step. In contrast, reverse-mode autodiff is just a technique to compute gradients efficiently, and it happens to be used by backpropagation.\n", + "9. Here is a list of all the hyperparameters you can tweak in a basic MLP: the number of hidden layers, the number of neurons in each hidden layer, and the activation function used in each hidden layer and in the output layer. In general, the ReLU activation function (or one of its variants; see Chapter 11) is a good default for the hidden layers. For the output layer, in general you will want the sigmoid activation function for binary classification, the softmax activation function for multiclass classification, or no activation function for regression. If the MLP overfits the training data, you can try reducing the number of hidden layers and reducing the number of neurons per hidden layer." ] }, { @@ -1622,7 +1907,14 @@ "cell_type": "markdown", "metadata": {}, "source": [ - "*Exercise: Train a deep MLP on the MNIST dataset (you can load it using `tf.keras.datasets.mnist.load_data()`. See if you can get over 98% precision. Try searching for the optimal learning rate by using the approach presented in this chapter (i.e., by growing the learning rate exponentially, plotting the loss, and finding the point where the loss shoots up). Try adding all the bells and whistles—save checkpoints, use early stopping, and plot learning curves using TensorBoard.*" + "*Exercise: Train a deep MLP on the MNIST dataset (you can load it using `tf.keras.datasets.mnist.load_data()`. See if you can get over 98% accuracy by manually tuning the hyperparameters. Try searching for the optimal learning rate by using the approach presented in this chapter (i.e., by growing the learning rate exponentially, plotting the loss, and finding the point where the loss shoots up). Next, try tuning the hyperparameters using Keras Tuner with all the bells and whistles—save checkpoints, use early stopping, and plot learning curves using TensorBoard.*" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "**TODO**: update this solution to use Keras Tuner." ] }, { @@ -1634,7 +1926,7 @@ }, { "cell_type": "code", - "execution_count": 107, + "execution_count": 108, "metadata": {}, "outputs": [], "source": [ @@ -1650,7 +1942,7 @@ }, { "cell_type": "code", - "execution_count": 108, + "execution_count": 109, "metadata": {}, "outputs": [], "source": [ @@ -1666,7 +1958,7 @@ }, { "cell_type": "code", - "execution_count": 109, + "execution_count": 110, "metadata": {}, "outputs": [], "source": [ @@ -1682,7 +1974,7 @@ }, { "cell_type": "code", - "execution_count": 110, + "execution_count": 111, "metadata": {}, "outputs": [], "source": [ @@ -1701,7 +1993,7 @@ }, { "cell_type": "code", - "execution_count": 111, + "execution_count": 112, "metadata": {}, "outputs": [], "source": [ @@ -1719,7 +2011,7 @@ }, { "cell_type": "code", - "execution_count": 112, + "execution_count": 113, "metadata": {}, "outputs": [], "source": [ @@ -1735,7 +2027,7 @@ }, { "cell_type": "code", - "execution_count": 113, + "execution_count": 114, "metadata": {}, "outputs": [], "source": [ @@ -1744,7 +2036,7 @@ }, { "cell_type": "code", - "execution_count": 114, + "execution_count": 115, "metadata": {}, "outputs": [], "source": [ @@ -1760,7 +2052,7 @@ }, { "cell_type": "code", - "execution_count": 115, + "execution_count": 116, "metadata": {}, "outputs": [], "source": [ @@ -1773,7 +2065,7 @@ " plt.subplot(n_rows, n_cols, index + 1)\n", " plt.imshow(X_train[index], cmap=\"binary\", interpolation=\"nearest\")\n", " plt.axis('off')\n", - " plt.title(y_train[index], fontsize=12)\n", + " plt.title(y_train[index])\n", "plt.subplots_adjust(wspace=0.2, hspace=0.5)\n", "plt.show()" ] @@ -1787,7 +2079,7 @@ }, { "cell_type": "code", - "execution_count": 116, + "execution_count": 117, "metadata": {}, "outputs": [], "source": [ @@ -1806,7 +2098,7 @@ }, { "cell_type": "code", - "execution_count": 117, + "execution_count": 118, "metadata": {}, "outputs": [], "source": [ @@ -1817,7 +2109,7 @@ }, { "cell_type": "code", - "execution_count": 118, + "execution_count": 119, "metadata": {}, "outputs": [], "source": [ @@ -1838,12 +2130,12 @@ }, { "cell_type": "code", - "execution_count": 119, + "execution_count": 120, "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=tf.keras.optimizers.SGD(learning_rate=1e-3),\n", + "optimizer = tf.keras.optimizers.SGD(learning_rate=1e-3)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer,\n", " metrics=[\"accuracy\"])\n", "expon_lr = ExponentialLearningRate(factor=1.005)" ] @@ -1857,7 +2149,7 @@ }, { "cell_type": "code", - "execution_count": 120, + "execution_count": 121, "metadata": {}, "outputs": [], "source": [ @@ -1875,7 +2167,7 @@ }, { "cell_type": "code", - "execution_count": 121, + "execution_count": 122, "metadata": {}, "outputs": [], "source": [ @@ -1897,7 +2189,7 @@ }, { "cell_type": "code", - "execution_count": 122, + "execution_count": 123, "metadata": {}, "outputs": [], "source": [ @@ -1908,7 +2200,7 @@ }, { "cell_type": "code", - "execution_count": 123, + "execution_count": 124, "metadata": {}, "outputs": [], "source": [ @@ -1922,18 +2214,18 @@ }, { "cell_type": "code", - "execution_count": 124, + "execution_count": 125, "metadata": {}, "outputs": [], "source": [ - "model.compile(loss=\"sparse_categorical_crossentropy\",\n", - " optimizer=tf.keras.optimizers.SGD(learning_rate=3e-1),\n", + "optimizer = tf.keras.optimizers.SGD(learning_rate=3e-1)\n", + "model.compile(loss=\"sparse_categorical_crossentropy\", optimizer=optimizer,\n", " metrics=[\"accuracy\"])" ] }, { "cell_type": "code", - "execution_count": 6, + "execution_count": 126, "metadata": {}, "outputs": [], "source": [ @@ -1944,7 +2236,7 @@ }, { "cell_type": "code", - "execution_count": 126, + "execution_count": 127, "metadata": {}, "outputs": [], "source": [ @@ -1959,7 +2251,7 @@ }, { "cell_type": "code", - "execution_count": 127, + "execution_count": 128, "metadata": {}, "outputs": [], "source": [ @@ -1976,11 +2268,11 @@ }, { "cell_type": "code", - "execution_count": 128, + "execution_count": 129, "metadata": {}, "outputs": [], "source": [ - "%tensorboard --logdir=./my_mnist_logs --port=6006" + "%tensorboard --logdir=./my_mnist_logs" ] }, { @@ -1993,7 +2285,7 @@ ], "metadata": { "kernelspec": { - "display_name": "Python 3 (ipykernel)", + "display_name": "Python 3", "language": "python", "name": "python3" },