From b60b2e0faf4ede575b84cd9f7259cbe4587ef1aa Mon Sep 17 00:00:00 2001 From: R-icntay <63848664+R-icntay@users.noreply.github.com> Date: Mon, 9 Aug 2021 16:55:20 +0300 Subject: [PATCH] Add R resources for lesson 08: logistic regression --- .../4-Logistic/images/r_learners_sm.jpeg | Bin 0 -> 40334 bytes .../4-Logistic/solution/lesson_4-R.ipynb | 751 ++++++++++++++++++ 2-Regression/4-Logistic/solution/lesson_4.Rmd | 430 ++++++++++ 3 files changed, 1181 insertions(+) create mode 100644 2-Regression/4-Logistic/images/r_learners_sm.jpeg create mode 100644 2-Regression/4-Logistic/solution/lesson_4-R.ipynb create mode 100644 2-Regression/4-Logistic/solution/lesson_4.Rmd diff --git a/2-Regression/4-Logistic/images/r_learners_sm.jpeg b/2-Regression/4-Logistic/images/r_learners_sm.jpeg new file mode 100644 index 0000000000000000000000000000000000000000..ff8d2945dc6ec7751987c5224f1d3139beae8467 GIT binary patch literal 40334 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 G&HWE+c~2Ms literal 0 HcmV?d00001 diff --git a/2-Regression/4-Logistic/solution/lesson_4-R.ipynb b/2-Regression/4-Logistic/solution/lesson_4-R.ipynb new file mode 100644 index 00000000..59468575 --- /dev/null +++ b/2-Regression/4-Logistic/solution/lesson_4-R.ipynb @@ -0,0 +1,751 @@ +{ + "nbformat": 4, + "nbformat_minor": 2, + "metadata": { + "colab": { + "name": "Untitled10.ipynb", + "provenance": [], + "collapsed_sections": [] + }, + "kernelspec": { + "name": "ir", + "display_name": "R" + }, + "language_info": { + "name": "R" + } + }, + "cells": [ + { + "cell_type": "markdown", + "source": [ + "# Build a regression model: logistic regression\n", + "
\n" + ], + "metadata": { + "id": "fVfEucLYkV9T" + } + }, + { + "cell_type": "markdown", + "source": [ + "## Build a logistic regression model - Lesson 4\r\n", + "\r\n", + "

\r\n", + " \r\n", + "

Infographic by Dasani Madipalli
\r\n", + "\r\n", + "" + ], + "metadata": { + "id": "QizKKpzakfx2" + } + }, + { + "cell_type": "markdown", + "source": [ + "#### ** [Pre-lecture quiz](https://jolly-sea-0a877260f.azurestaticapps.net/quiz/15/)**\n", + "\n", + "#### Introduction\n", + "\n", + "In this final lesson on Regression, one of the basic *classic* ML techniques, we will take a look at Logistic Regression. You would use this technique to discover patterns to predict `binary` `categories`. Is this candy chocolate or not? Is this disease contagious or not? Will this customer choose this product or not?\n", + "\n", + "In this lesson, you will learn:\n", + "\n", + "- Techniques for logistic regression\n", + "\n", + "โœ… Deepen your understanding of working with this type of regression in this [Learn module](https://docs.microsoft.com/learn/modules/train-evaluate-classification-models?WT.mc_id=academic-15963-cxa)\n", + "\n", + "#### **Prerequisite**\n", + "\n", + "Having worked with the pumpkin data, we are now familiar enough with it to realize that there's one binary category that we can work with: `Color`.\n", + "\n", + "Let's build a logistic regression model to predict that, given some variables, *what color a given pumpkin is likely to be* (orange ๐ŸŽƒ or white ๐Ÿ‘ป).\n", + "\n", + "> Why are we talking about binary classification in a lesson grouping about regression? Only for linguistic convenience, as logistic regression is [really a classification method](https://scikit-learn.org/stable/modules/linear_model.html#logistic-regression), albeit a linear-based one. Learn about other ways to classify data in the next lesson group.\n", + "\n", + "For this lesson, we'll require the following packages:\n", + "\n", + "- `tidyverse`: The [tidyverse](https://www.tidyverse.org/) is a [collection of R packages](https://www.tidyverse.org/packages) designed to makes data science faster, easier and more fun!\n", + "\n", + "- `tidymodels`: The [tidymodels](https://www.tidymodels.org/) framework is a [collection of packages](https://www.tidymodels.org/packages/) for modeling and machine learning.\n", + "\n", + "- `janitor`: The [janitor package](https://github.com/sfirke/janitor) provides simple little tools for examining and cleaning dirty data.\n", + "\n", + "- `ggbeeswarm`: The [ggbeeswarm package](https://github.com/eclarke/ggbeeswarm) provides methods to create beeswarm-style plots using ggplot2.\n", + "\n", + "You can have them installed as:\n", + "\n", + "`install.packages(c(\"tidyverse\", \"tidymodels\", \"janitor\", \"ggbeeswarm\"))`\n", + "\n", + "Alternatiely, the script below checks whether you have the packages required to complete this module and installs them for you in case they are missing." + ], + "metadata": { + "id": "KPmut75XkmXY" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "suppressWarnings(if (!require(\"pacman\")) install.packages(\"pacman\"))\r\n", + "\r\n", + "pacman::p_load(tidyverse, tidymodels, janitor, ggbeeswarm)" + ], + "outputs": [], + "metadata": { + "id": "dnIGNNttkx_O" + } + }, + { + "cell_type": "markdown", + "source": [ + "## ** Define the question**\r\n", + "\r\n", + "For our purposes, we will express this as a binary: 'Orange' or 'Not Orange'. There is also a 'striped' category in our dataset but there are few instances of it, so we will not use it. It disappears once we remove null values from the dataset, anyway.\r\n", + "\r\n", + "> ๐ŸŽƒ Fun fact, we sometimes call white pumpkins 'ghost' pumpkins. They aren't very easy to carve, so they aren't as popular as the orange ones but they are cool looking!\r\n", + "\r\n", + "## **About logistic regression**\r\n", + "\r\n", + "Logistic regression differs from linear regression, which you learned about previously, in a few important ways.\r\n", + "\r\n", + "#### **Binary classification**\r\n", + "\r\n", + "Logistic regression does not offer the same features as linear regression. The former offers a prediction about a `binary category` (\"orange or not orange\") whereas the latter is capable of predicting `continual values`, for example given the origin of a pumpkin and the time of harvest, *how much its price will rise*.\r\n", + "\r\n", + "

\r\n", + " \r\n", + "

Infographic by Dasani Madipalli
\r\n", + "\r\n", + "" + ], + "metadata": { + "id": "ws-hP_SXk2O6" + } + }, + { + "cell_type": "markdown", + "source": [ + "#### **Other classifications**\r\n", + "\r\n", + "There are other types of logistic regression, including multinomial and ordinal:\r\n", + "\r\n", + "- **Multinomial**, which involves having more than one category - \"Orange, White, and Striped\".\r\n", + "\r\n", + "- **Ordinal**, which involves ordered categories, useful if we wanted to order our outcomes logically, like our pumpkins that are ordered by a finite number of sizes (mini,sm,med,lg,xl,xxl).\r\n", + "\r\n", + "

\r\n", + " \r\n", + "

Infographic by Dasani Madipalli
\r\n", + "\r\n", + "" + ], + "metadata": { + "id": "LkLN-ZgDlBEc" + } + }, + { + "cell_type": "markdown", + "source": [ + "**It's still linear**\n", + "\n", + "Even though this type of Regression is all about 'category predictions', it still works best when there is a clear linear relationship between the dependent variable (color) and the other independent variables (the rest of the dataset, like city name and size). It's good to get an idea of whether there is any linearity dividing these variables or not.\n", + "\n", + "#### **Variables DO NOT have to correlate**\n", + "\n", + "Remember how linear regression worked better with more correlated variables? Logistic regression is the opposite - the variables don't have to align. That works for this data which has somewhat weak correlations.\n", + "\n", + "#### **You need a lot of clean data**\n", + "\n", + "Logistic regression will give more accurate results if you use more data; our small dataset is not optimal for this task, so keep that in mind.\n", + "\n", + "โœ… Think about the types of data that would lend themselves well to logistic regression\n" + ], + "metadata": { + "id": "D8_JoVZtlHUt" + } + }, + { + "cell_type": "markdown", + "source": [ + "## 1. Tidy the data\n", + "\n", + "Now, the fun begins! Let's start by importing the data, cleaning the data a bit, dropping rows containing missing values and selecting only some of the columns:" + ], + "metadata": { + "id": "LPj8Ib1AlIua" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Load the core tidyverse packages\r\n", + "library(tidyverse)\r\n", + "\r\n", + "# Import the data and clean column names\r\n", + "pumpkins <- read_csv(file = \"https://raw.githubusercontent.com/microsoft/ML-For-Beginners/main/2-Regression/data/US-pumpkins.csv\") %>% \r\n", + " clean_names()\r\n", + "\r\n", + "# Select desired columns\r\n", + "pumpkins_select <- pumpkins %>% \r\n", + " select(c(city_name, package, variety, origin, item_size, color)) \r\n", + "\r\n", + "# Drop rows containing missing values and encode color as factor (category)\r\n", + "pumpkins_select <- pumpkins_select %>% \r\n", + " drop_na() %>% \r\n", + " mutate(color = factor(color))\r\n", + "\r\n", + "# View the first few rows\r\n", + "pumpkins_select %>% \r\n", + " slice_head(n = 5)\r\n" + ], + "outputs": [], + "metadata": { + "id": "Q8oKJ8PAlLM0" + } + }, + { + "cell_type": "markdown", + "source": [ + "Sometimes, we may want some little more information on our data. We can have a look at the `data`, `its structure` and the `data type` of its features by using the [*glimpse()*](https://pillar.r-lib.org/reference/glimpse.html) function as below:" + ], + "metadata": { + "id": "tKY5eN8alPNn" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "pumpkins_select %>% \r\n", + " glimpse()" + ], + "outputs": [], + "metadata": { + "id": "wDpatL1WlShu" + } + }, + { + "cell_type": "markdown", + "source": [ + "Wow! Seems that all our columns are all of type *character*, further alluding that they are all categorical.\n", + "\n", + "Let's confirm that we will actually be doing a binary classification problem:" + ], + "metadata": { + "id": "QbdC2b0JlU2G" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Subset distinct observations in outcome column\r\n", + "pumpkins_select %>% \r\n", + " distinct(color)" + ], + "outputs": [], + "metadata": { + "id": "Gys-Q18rlZpE" + } + }, + { + "cell_type": "markdown", + "source": [ + "๐Ÿฅณ๐Ÿฅณ That went down well!\n", + "\n", + "## 2. Explore the data\n", + "\n", + "The goal of data exploration is to try to understand the `relationships` between its attributes; in particular, any apparent correlation between the *features* and the *label* your model will try to predict. One way of doing this is by using data visualization.\n", + "\n", + "Given our the data types of our columns, we can `encode` them and be on our way to making some visualizations. This simply involves `translating` a column with `categorical values` for example our columns of type *char*, into one or more `numeric columns` that take the place of the original. - Something we did in our [last lesson](https://github.com/microsoft/ML-For-Beginners/blob/main/2-Regression/3-Linear/solution/lesson_3-R.ipynb).\n", + "\n", + "Tidymodels provides yet another neat package: [recipes](https://recipes.tidymodels.org/)- a package for preprocessing data. We'll define a `recipe` that specifies that all predictor columns should be encoded into a set of integers , `prep` it to estimates the required quantities and statistics needed by any operations and finally `bake` to apply the computations to new data.\n", + "\n", + "> Normally, recipes is usually used as a preprocessor for modelling where it defines what steps should be applied to a data set in order to get it ready for modelling. In that case it is **highly recommend** that you use a `workflow()` instead of manually estimating a recipe using prep and bake. We'll see all this in just a moment.\n", + ">\n", + "> However for now, we are using recipes + prep + bake to specify what steps should be applied to a data set in order to get it ready for data analysis and then extract the preprocessed data with the steps applied." + ], + "metadata": { + "id": "kn_20wSPldVH" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Preprocess and extract data to allow some data analysis\r\n", + "baked_pumpkins <- recipe(color ~ ., data = pumpkins_select) %>% \r\n", + " # Encode all columns to a set of integers\r\n", + " step_integer(all_predictors(), zero_based = T) %>% \r\n", + " prep() %>% \r\n", + " bake(new_data = NULL)\r\n", + "\r\n", + "\r\n", + "# Display the first few rows of preprocessed data\r\n", + "baked_pumpkins %>% \r\n", + " slice_head(n = 5)" + ], + "outputs": [], + "metadata": { + "id": "syaCgFQ_lijg" + } + }, + { + "cell_type": "markdown", + "source": [ + "Now let's compare the feature distributions for each label value using box plots. We'll begin by formatting the data to a *long* format to make it somewhat easier to make multiple `facets`." + ], + "metadata": { + "id": "RlkOZ_C5lldq" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Pivot data to long format\r\n", + "baked_pumpkins_long <- baked_pumpkins %>% \r\n", + " pivot_longer(!color, names_to = \"features\", values_to = \"values\")\r\n", + "\r\n", + "\r\n", + "# Print out restructured data\r\n", + "baked_pumpkins_long %>% \r\n", + " slice_head(n = 10)\r\n" + ], + "outputs": [], + "metadata": { + "id": "putq8DagltUQ" + } + }, + { + "cell_type": "markdown", + "source": [ + "Now, let's make some boxplots showing the distribution of the predictors with respect to the outcome color." + ], + "metadata": { + "id": "-RHm-12zlt-B" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "theme_set(theme_light())\r\n", + "#Make a box plot for each predictor feature\r\n", + "baked_pumpkins_long %>% \r\n", + " mutate(color = factor(color)) %>% \r\n", + " ggplot(mapping = aes(x = color, y = values, fill = features)) +\r\n", + " geom_boxplot() + \r\n", + " facet_wrap(~ features, scales = \"free\", ncol = 3) +\r\n", + " scale_color_viridis_d(option = \"cividis\", end = .8) +\r\n", + " theme(legend.position = \"none\")" + ], + "outputs": [], + "metadata": { + "id": "3Py4i1p1l3hP" + } + }, + { + "cell_type": "markdown", + "source": [ + "Amazing๐Ÿคฉ! For some of the features, there's a noticeable difference in the distribution for each color label. For instance, it seems the white pumpkins can be found in smaller packages and in some particular varieties of pumpkins. The *item_size* category also seems to make a difference in the color distribution. These features may help predict the color of a pumpkin.\n", + "\n", + "#### **Use a swarm plot**\n", + "\n", + "Color is a binary category (Orange or Not), it's called `categorical data`. There are other various ways of [visualizing categorical data](https://seaborn.pydata.org/tutorial/categorical.html?highlight=bar).\n", + "\n", + "Try a `swarm plot` to show the distribution of color with respect to the item_size.\n", + "\n", + "We'll use the [ggbeeswarm package](https://github.com/eclarke/ggbeeswarm) which provides methods to create beeswarm-style plots using ggplot2. Beeswarm plots are a way of plotting points that would ordinarily overlap so that they fall next to each other instead." + ], + "metadata": { + "id": "2LSj6_LCl68V" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Create beeswarm plots of color and item_size\r\n", + "baked_pumpkins %>% \r\n", + " mutate(color = factor(color)) %>% \r\n", + " ggplot(mapping = aes(x = color, y = item_size, color = color)) +\r\n", + " geom_quasirandom() +\r\n", + " scale_color_brewer(palette = \"Dark2\", direction = -1) +\r\n", + " theme(legend.position = \"none\")" + ], + "outputs": [], + "metadata": { + "id": "hGKeRgUemMTb" + } + }, + { + "cell_type": "markdown", + "source": [ + "#### **Violin plot**\n", + "\n", + "A 'violin' type plot is useful as you can easily visualize the way that data in the two categories is distributed. [`Violin plots`](https://en.wikipedia.org/wiki/Violin_plot) are similar to box plots, except that they also show the probability density of the data at different values. Violin plots don't work so well with smaller datasets as the distribution is displayed more 'smoothly'." + ], + "metadata": { + "id": "_9wdZJH5mOvN" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Create a violin plot of color and item_size\r\n", + "baked_pumpkins %>%\r\n", + " mutate(color = factor(color)) %>% \r\n", + " ggplot(mapping = aes(x = color, y = item_size, fill = color)) +\r\n", + " geom_violin() +\r\n", + " geom_boxplot(color = \"black\", fill = \"white\", width = 0.02) +\r\n", + " scale_fill_brewer(palette = \"Dark2\", direction = -1) +\r\n", + " theme(legend.position = \"none\")" + ], + "outputs": [], + "metadata": { + "id": "LFFFymujmTAZ" + } + }, + { + "cell_type": "markdown", + "source": [ + "Now that we have an idea of the relationship between the binary categories of color and the larger group of sizes, let's explore logistic regression to determine a given pumpkin's likely color.\r\n", + "\r\n", + "## 3. Build your logistic regression model\r\n", + "\r\n", + "

\r\n", + " \r\n", + "

Infographic by Dasani Madipalli
\r\n", + "\r\n", + "> **๐Ÿงฎ Show Me The Math**\r\n", + ">\r\n", + "> Remember how `linear regression` often used `ordinary least squares` to arrive at a value? `Logistic regression` relies on the concept of 'maximum likelihood' using [`sigmoid functions`](https://wikipedia.org/wiki/Sigmoid_function). A Sigmoid Function on a plot looks like an `S shape`. It takes a value and maps it to somewhere between 0 and 1. Its curve is also called a 'logistic curve'. Its formula looks like this:\r\n", + ">\r\n", + "> \r\n", + "

\r\n", + " where the sigmoid's midpoint finds itself at x's 0 point, L is the curve's maximum value, and k is the curve's steepness. If the outcome of the function is more than 0.5, the label in question will be given the class 1 of the binary choice. If not, it will be classified as 0.\r\n", + "\r\n", + "Let's begin by splitting the data into `training` and `test` sets. The training set is used to train a classifier so that it finds a statistical relationship between the features and the label value.\r\n", + "\r\n", + "It is best practice to hold out some of your data for **testing** in order to get a better estimate of how your models will perform on new data by comparing the predicted labels with the already known labels in the test set. [rsample](https://rsample.tidymodels.org/), a package in Tidymodels, provides infrastructure for efficient data splitting and resampling:" + ], + "metadata": { + "id": "RA_bnMS9mVo8" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Split data into 80% for training and 20% for testing\r\n", + "set.seed(2056)\r\n", + "pumpkins_split <- pumpkins_select %>% \r\n", + " initial_split(prop = 0.8)\r\n", + "\r\n", + "# Extract the data in each split\r\n", + "pumpkins_train <- training(pumpkins_split)\r\n", + "pumpkins_test <- testing(pumpkins_split)\r\n", + "\r\n", + "# Print out the first 5 rows of the training set\r\n", + "pumpkins_train %>% \r\n", + " slice_head(n = 5)" + ], + "outputs": [], + "metadata": { + "id": "PQdpEYYPmdGW" + } + }, + { + "cell_type": "markdown", + "source": [ + "๐Ÿ™Œ We are now ready to train a model by fitting the training features to the training label (color).\n", + "\n", + "We'll begin by creating a recipe that specifies the preprocessing steps that should be carried out on our data to get it ready for modelling i.e: encoding categorical variables into a set of integers.\n", + "\n", + "There are quite a number of ways to specify a logistic regression model in Tidymodels. See `?logistic_reg()` For now, we'll specify a logistic regression model via the default `stats::glm()` engine." + ], + "metadata": { + "id": "MX9LipSimhn0" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Create a recipe that specifies preprocessing steps for modelling\r\n", + "pumpkins_recipe <- recipe(color ~ ., data = pumpkins_train) %>% \r\n", + " step_integer(all_predictors(), zero_based = TRUE)\r\n", + "\r\n", + "\r\n", + "# Create a logistic model specification\r\n", + "log_reg <- logistic_reg() %>% \r\n", + " set_engine(\"glm\") %>% \r\n", + " set_mode(\"classification\")\r\n" + ], + "outputs": [], + "metadata": { + "id": "0Eo5-SbSmm2-" + } + }, + { + "cell_type": "markdown", + "source": [ + "Now that we have a recipe and a model specification, we need to find a way of bundling them together into an object that will first preprocess the data (prep+bake behind the scenes), fit the model on the preprocessed data and also allow for potential post-processing activities.\n", + "\n", + "In Tidymodels, this convenient object is called a [`workflow`](https://workflows.tidymodels.org/) and conveniently holds your modeling components." + ], + "metadata": { + "id": "G599GKhXmqWf" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Bundle modelling components in a workflow\r\n", + "log_reg_wf <- workflow() %>% \r\n", + " add_recipe(pumpkins_recipe) %>% \r\n", + " add_model(log_reg)\r\n", + "\r\n", + "# Print out the workflow\r\n", + "log_reg_wf\r\n" + ], + "outputs": [], + "metadata": { + "id": "cRoU0tpbmu1T" + } + }, + { + "cell_type": "markdown", + "source": [ + "After a workflow has been *specified*, a model can be `trained` using the [`fit()`](https://tidymodels.github.io/parsnip/reference/fit.html) function. The workflow will estimate a recipe and preprocess the data before training, so we won't have to manually do that using prep and bake." + ], + "metadata": { + "id": "JnRXKmREnEpd" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Train the model\r\n", + "wf_fit <- log_reg_wf %>% \r\n", + " fit(data = pumpkins_train)\r\n", + "\r\n", + "# Print the trained workflow\r\n", + "wf_fit" + ], + "outputs": [], + "metadata": { + "id": "ehFwfkjWnNCb" + } + }, + { + "cell_type": "markdown", + "source": [ + "The model print out shows the coefficients learned during training.\n", + "\n", + "Now we've trained the model using the training data, we can make predictions on the test data using [parsnip::predict()](https://parsnip.tidymodels.org/reference/predict.model_fit.html). Let's start by using the model to predict labels for our test set and the probabilities for each label. When the probability is more than 0.5, the predict class is `ORANGE` else `WHITE`." + ], + "metadata": { + "id": "w01dGNZjnOJQ" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Make predictions for color and corresponding probabilities\r\n", + "results <- pumpkins_test %>% select(color) %>% \r\n", + " bind_cols(wf_fit %>% \r\n", + " predict(new_data = pumpkins_test)) %>%\r\n", + " bind_cols(wf_fit %>%\r\n", + " predict(new_data = pumpkins_test, type = \"prob\"))\r\n", + "\r\n", + "# Compare predictions\r\n", + "results %>% \r\n", + " slice_head(n = 10)" + ], + "outputs": [], + "metadata": { + "id": "K8PNjPfTnak2" + } + }, + { + "cell_type": "markdown", + "source": [ + "Very nice! This provides some more insights into how logistic regression works.\n", + "\n", + "Comparing each prediction with its corresponding \"ground truth\" actual value isn't a very efficient way to determine how well the model is predicting. Fortunately, Tidymodels has a few more tricks up its sleeve: [`yardstick`](https://yardstick.tidymodels.org/) - a package used to measure the effectiveness of models using performance metrics.\n", + "\n", + "One performance metric associated with classification problems is the [`confusion matrix`](https://wikipedia.org/wiki/Confusion_matrix). A confusion matrix describes how well a classification model performs. A confusion matrix tabulates how many examples in each class were correctly classified by a model. In our case, it will show you how many orange pumpkins were classified as orange and how many white pumpkins were classified as white; the confusion matrix also shows you how many were classified into the **wrong** categories.\n", + "\n", + "The [**`conf_mat()`**](https://tidymodels.github.io/yardstick/reference/conf_mat.html) function from yardstick calculates this cross-tabulation of observed and predicted classes." + ], + "metadata": { + "id": "N3J-yW0wngKo" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Confusion matrix for prediction results\r\n", + "conf_mat(data = results, truth = color, estimate = .pred_class)" + ], + "outputs": [], + "metadata": { + "id": "0RD77Dq1nl2j" + } + }, + { + "cell_type": "markdown", + "source": [ + "Let's interpret the confusion matrix. Our model is asked to classify pumpkins between two binary categories, category `orange` and category `not-orange`\n", + "\n", + "- If your model predicts a pumpkin as orange and it belongs to category 'orange' in reality we call it a `true positive`, shown by the top left number.\n", + "\n", + "- If your model predicts a pumpkin as not orange and it belongs to category 'orange' in reality we call it a `false negative`, shown by the bottom left number.\n", + "\n", + "- If your model predicts a pumpkin as orange and it belongs to category 'not-orange' in reality we call it a `false positive`, shown by the top right number.\n", + "\n", + "- If your model predicts a pumpkin as not orange and it belongs to category 'not-orange' in reality we call it a `true negative`, shown by the bottom right number.\n", + "\n", + "\n", + "| **Truth** |\n", + "|:-----:|\n", + "\n", + "\n", + "| | | |\n", + "|---------------|--------|-------|\n", + "| **Predicted** | ORANGE | WHITE |\n", + "| ORANGE | TP | FP |\n", + "| WHITE | FN | TN |" + ], + "metadata": { + "id": "H61sFwdOnoiO" + } + }, + { + "cell_type": "markdown", + "source": [ + "As you might have guessed it's preferable to have a larger number of true positives and true negatives and a lower number of false positives and false negatives, which implies that the model performs better.\n", + "\n", + "The confusion matrix is helpful since it gives rise to other metrics that can help us better evaluate the performance of a classification model. Let's go through some of them:\n", + "\n", + "๐ŸŽ“ Precision: `TP/(TP + FP)` defined as the proportion of predicted positives that are actually positive. Also called [positive predictive value](https://en.wikipedia.org/wiki/Positive_predictive_value \"Positive predictive value\")\n", + "\n", + "๐ŸŽ“ Recall: `TP/(TP + FN)` defined as the proportion of positive results out of the number of samples which were actually positive. Also known as `sensitivity`.\n", + "\n", + "๐ŸŽ“ Specificity: `TN/(TN + FP)` defined as the proportion of negative results out of the number of samples which were actually negative.\n", + "\n", + "๐ŸŽ“ Accuracy: `TP + TN/(TP + TN + FP + FN)` The percentage of labels predicted accurately for a sample.\n", + "\n", + "๐ŸŽ“ F Measure: A weighted average of the precision and recall, with best being 1 and worst being 0.\n", + "\n", + "Let's calculate these metrics!" + ], + "metadata": { + "id": "Yc6QUie2oQUr" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Combine metric functions and calculate them all at once\r\n", + "eval_metrics <- metric_set(ppv, recall, spec, f_meas, accuracy)\r\n", + "eval_metrics(data = results, truth = color, estimate = .pred_class)" + ], + "outputs": [], + "metadata": { + "id": "p6rXx_T3oVxX" + } + }, + { + "cell_type": "markdown", + "source": [ + "#### **Visualize the ROC curve of this model**\n", + "\n", + "For a start, this is not a bad model; its precision, recall, F measure and accuracy are in the 80% range so ideally you could use it to predict the color of a pumpkin given a set of variables. It also seems that our model was not really able to identify the white pumpkins ๐Ÿง. Could you guess why? One reason could be because of the high prevalence of ORANGE pumpkins in our training set making our model more inclined to predict the majority class.\n", + "\n", + "Let's do one more visualization to see the so-called [`ROC score`](https://en.wikipedia.org/wiki/Receiver_operating_characteristic):" + ], + "metadata": { + "id": "JcenzZo1oaKR" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Make a roc_curve\r\n", + "results %>% \r\n", + " roc_curve(color, .pred_ORANGE) %>% \r\n", + " autoplot()" + ], + "outputs": [], + "metadata": { + "id": "BcmkHHHwogRB" + } + }, + { + "cell_type": "markdown", + "source": [ + "ROC curves are often used to get a view of the output of a classifier in terms of its true vs. false positives. ROC curves typically feature `True Positive Rate`/Sensitivity on the Y axis, and `False Positive Rate`/1-Specificity on the X axis. Thus, the steepness of the curve and the space between the midpoint line and the curve matter: you want a curve that quickly heads up and over the line. In our case, there are false positives to start with, and then the line heads up and over properly.\n", + "\n", + "Finally, let's use `yardstick::roc_auc()` to calculate the actual Area Under the Curve. One way of interpreting AUC is as the probability that the model ranks a random positive example more highly than a random negative example." + ], + "metadata": { + "id": "P_an3vc1oqjI" + } + }, + { + "cell_type": "code", + "execution_count": null, + "source": [ + "# Calculate area under curve\r\n", + "results %>% \r\n", + " roc_auc(color, .pred_ORANGE)" + ], + "outputs": [], + "metadata": { + "id": "SZyy5BT8ovew" + } + }, + { + "cell_type": "markdown", + "source": [ + "The result is around `0.67053`. Given that the AUC ranges from 0 to 1, you want a big score, since a model that is 100% correct in its predictions will have an AUC of 1; in this case, the model is *pretty good*.\r\n", + "\r\n", + "In future lessons on classifications, you will learn how to improve your model's scores (such as dealing with imbalanced data in this case).\r\n", + "\r\n", + "But for now, congratulations ๐ŸŽ‰๐ŸŽ‰๐ŸŽ‰! You've completed these regression lessons!\r\n", + "\r\n", + "You R awesome!\r\n", + "\r\n", + "

\r\n", + " \r\n", + "

Artwork by @allison_horst
\r\n", + "\r\n", + "\r\n" + ], + "metadata": { + "id": "5jtVKLTVoy6u" + } + } + ] +} \ No newline at end of file diff --git a/2-Regression/4-Logistic/solution/lesson_4.Rmd b/2-Regression/4-Logistic/solution/lesson_4.Rmd new file mode 100644 index 00000000..8d99dc62 --- /dev/null +++ b/2-Regression/4-Logistic/solution/lesson_4.Rmd @@ -0,0 +1,430 @@ +--- +title: 'Build a regression model: logistic regression' +output: + html_document: + df_print: paged + theme: flatly + highlight: breezedark + toc: yes + toc_float: yes + code_download: yes +--- + +## Build a logistic regression model - Lesson 4 + +![Infographic by Dasani Madipalli](../images/logistic-linear.png){width="600"} + +#### ** [Pre-lecture quiz](https://jolly-sea-0a877260f.azurestaticapps.net/quiz/15/)** + +#### Introduction + +In this final lesson on Regression, one of the basic *classic* ML techniques, we will take a look at Logistic Regression. You would use this technique to discover patterns to predict `binary` `categories`. Is this candy chocolate or not? Is this disease contagious or not? Will this customer choose this product or not? + +In this lesson, you will learn: + +- Techniques for logistic regression + +โœ… Deepen your understanding of working with this type of regression in this [Learn module](https://docs.microsoft.com/learn/modules/train-evaluate-classification-models?WT.mc_id=academic-15963-cxa) + +#### **Prerequisite** + +Having worked with the pumpkin data, we are now familiar enough with it to realize that there's one binary category that we can work with: `Color`. + +Let's build a logistic regression model to predict that, given some variables, *what color a given pumpkin is likely to be* (orange ๐ŸŽƒ or white ๐Ÿ‘ป). + +> Why are we talking about binary classification in a lesson grouping about regression? Only for linguistic convenience, as logistic regression is [really a classification method](https://scikit-learn.org/stable/modules/linear_model.html#logistic-regression), albeit a linear-based one. Learn about other ways to classify data in the next lesson group. + +For this lesson, we'll require the following packages: + +- `tidyverse`: The [tidyverse](https://www.tidyverse.org/) is a [collection of R packages](https://www.tidyverse.org/packages) designed to makes data science faster, easier and more fun! + +- `tidymodels`: The [tidymodels](https://www.tidymodels.org/) framework is a [collection of packages](https://www.tidymodels.org/packages/) for modeling and machine learning. + +- `janitor`: The [janitor package](https://github.com/sfirke/janitor) provides simple little tools for examining and cleaning dirty data. + +- `ggbeeswarm`: The [ggbeeswarm package](https://github.com/eclarke/ggbeeswarm) provides methods to create beeswarm-style plots using ggplot2. + +You can have them installed as: + +`install.packages(c("tidyverse", "tidymodels", "janitor", "ggbeeswarm"))` + +Alternatiely, the script below checks whether you have the packages required to complete this module and installs them for you in case they are missing. + +```{r, message=F, warning=F} +suppressWarnings(if (!require("pacman"))install.packages("pacman")) + +pacman::p_load(tidyverse, tidymodels, janitor, ggbeeswarm) +``` + +## ** Define the question** + +For our purposes, we will express this as a binary: 'Orange' or 'Not Orange'. There is also a 'striped' category in our dataset but there are few instances of it, so we will not use it. It disappears once we remove null values from the dataset, anyway. + +> ๐ŸŽƒ Fun fact, we sometimes call white pumpkins 'ghost' pumpkins. They aren't very easy to carve, so they aren't as popular as the orange ones but they are cool looking! + +## **About logistic regression** + +Logistic regression differs from linear regression, which you learned about previously, in a few important ways. + +#### **Binary classification** + +Logistic regression does not offer the same features as linear regression. The former offers a prediction about a `binary category` ("orange or not orange") whereas the latter is capable of predicting `continual values`, for example given the origin of a pumpkin and the time of harvest, *how much its price will rise*. + +![Infographic by Dasani Madipalli](../images/pumpkin-classifier.png){width="600"} + +#### **Other classifications** + +There are other types of logistic regression, including multinomial and ordinal: + +- **Multinomial**, which involves having more than one category - "Orange, White, and Striped". + +- **Ordinal**, which involves ordered categories, useful if we wanted to order our outcomes logically, like our pumpkins that are ordered by a finite number of sizes (mini,sm,med,lg,xl,xxl). + +![Infographic by Dasani Madipalli](../images/multinomial-ordinal.png){width="600"} + +\ +**It's still linear** + +Even though this type of Regression is all about 'category predictions', it still works best when there is a clear linear relationship between the dependent variable (color) and the other independent variables (the rest of the dataset, like city name and size). It's good to get an idea of whether there is any linearity dividing these variables or not. + +#### **Variables DO NOT have to correlate** + +Remember how linear regression worked better with more correlated variables? Logistic regression is the opposite - the variables don't have to align. That works for this data which has somewhat weak correlations. + +#### **You need a lot of clean data** + +Logistic regression will give more accurate results if you use more data; our small dataset is not optimal for this task, so keep that in mind. + +โœ… Think about the types of data that would lend themselves well to logistic regression + +## 1. Tidy the data + +Now, the fun begins! Let's start by importing the data, cleaning the data a bit, dropping rows containing missing values and selecting only some of the columns: + +```{r, tidyr, message=F, warning=F} +# Load the core tidyverse packages +library(tidyverse) + +# Import the data and clean column names +pumpkins <- read_csv(file = "https://raw.githubusercontent.com/microsoft/ML-For-Beginners/main/2-Regression/data/US-pumpkins.csv") %>% + clean_names() + +# Select desired columns +pumpkins_select <- pumpkins %>% + select(c(city_name, package, variety, origin, item_size, color)) + +# Drop rows containing missing values and encode color as factor (category) +pumpkins_select <- pumpkins_select %>% + drop_na() %>% + mutate(color = factor(color)) + +# View the first few rows +pumpkins_select %>% + slice_head(n = 5) + +``` + +Sometimes, we may want some little more information on our data. We can have a look at the `data`, `its structure` and the `data type` of its features by using the [*glimpse()*](https://pillar.r-lib.org/reference/glimpse.html) function as below: + +```{r glimpse} +pumpkins_select %>% + glimpse() +``` + +Wow! Seems that all our columns are all of type *character*, further alluding that they are all categorical. + +Let's confirm that we will actually be doing a binary classification problem: + +```{r distinct color} +# Subset distinct observations in outcome column +pumpkins_select %>% + distinct(color) + +``` + +๐Ÿฅณ๐Ÿฅณ That went down well! + +## 2. Explore the data + +The goal of data exploration is to try to understand the `relationships` between its attributes; in particular, any apparent correlation between the *features* and the *label* your model will try to predict. One way of doing this is by using data visualization. + +Given our the data types of our columns, we can `encode` them and be on our way to making some visualizations. This simply involves `translating` a column with `categorical values` for example our columns of type *char*, into one or more `numeric columns` that take the place of the original. - Something we did in our [last lesson](https://github.com/microsoft/ML-For-Beginners/blob/main/2-Regression/3-Linear/solution/lesson_3-R.ipynb). + +Tidymodels provides yet another neat package: [recipes](https://recipes.tidymodels.org/)- a package for preprocessing data. We'll define a `recipe` that specifies that all predictor columns should be encoded into a set of integers , `prep` it to estimates the required quantities and statistics needed by any operations and finally `bake` to apply the computations to new data. + +> Normally, recipes is usually used as a preprocessor for modelling where it defines what steps should be applied to a data set in order to get it ready for modelling. In that case it is **highly recommend** that you use a `workflow()` instead of manually estimating a recipe using prep and bake. We'll see all this in just a moment. +> +> However for now, we are using recipes + prep + bake to specify what steps should be applied to a data set in order to get it ready for data analysis and then extract the preprocessed data with the steps applied. + +```{r recipe_prep_bake} +# Preprocess and extract data to allow some data analysis +baked_pumpkins <- recipe(color ~ ., data = pumpkins_select) %>% + # Encode all columns to a set of integers + step_integer(all_predictors(), zero_based = T) %>% + prep() %>% + bake(new_data = NULL) + + +# Display the first few rows of preprocessed data +baked_pumpkins %>% + slice_head(n = 5) + +``` + +Now let's compare the feature distributions for each label value using box plots. We'll begin by formatting the data to a *long* format to make it somewhat easier to make multiple `facets`. + +```{r pivot} +# Pivot data to long format +baked_pumpkins_long <- baked_pumpkins %>% + pivot_longer(!color, names_to = "features", values_to = "values") + + +# Print out restructured data +baked_pumpkins_long %>% + slice_head(n = 10) + +``` + + +Now, let's make some boxplots showing the distribution of the predictors with respect to the outcome color! + +```{r boxplots} +theme_set(theme_light()) +#Make a box plot for each predictor feature +baked_pumpkins_long %>% + mutate(color = factor(color)) %>% + ggplot(mapping = aes(x = color, y = values, fill = features)) + + geom_boxplot() + + facet_wrap(~ features, scales = "free", ncol = 3) + + scale_color_viridis_d(option = "cividis", end = .8) + + theme(legend.position = "none") +``` + +Amazing๐Ÿคฉ! For some of the features, there's a noticeable difference in the distribution for each color label. For instance, it seems the white pumpkins can be found in smaller packages and in some particular varieties of pumpkins. The *item_size* category also seems to make a difference in the color distribution. These features may help predict the color of a pumpkin. + +#### **Use a swarm plot** + +Color is a binary category (Orange or Not), it's called `categorical data`. There are other various ways of [visualizing categorical data](https://seaborn.pydata.org/tutorial/categorical.html?highlight=bar). + +Try a `swarm plot` to show the distribution of color with respect to the item_size. + +We'll use the [ggbeeswarm package](https://github.com/eclarke/ggbeeswarm) which provides methods to create beeswarm-style plots using ggplot2. Beeswarm plots are a way of plotting points that would ordinarily overlap so that they fall next to each other instead. + +```{r bee_swarm plot} +# Create beeswarm plots of color and item_size +baked_pumpkins %>% + mutate(color = factor(color)) %>% + ggplot(mapping = aes(x = color, y = item_size, color = color)) + + geom_quasirandom() + + scale_color_brewer(palette = "Dark2", direction = -1) + + theme(legend.position = "none") +``` + +#### **Violin plot** + +A 'violin' type plot is useful as you can easily visualize the way that data in the two categories is distributed. [`Violin plots`](https://en.wikipedia.org/wiki/Violin_plot) are similar to box plots, except that they also show the probability density of the data at different values. Violin plots don't work so well with smaller datasets as the distribution is displayed more 'smoothly'. + +```{r violin_plot} +# Create a violin plot of color and item_size +baked_pumpkins %>% + mutate(color = factor(color)) %>% + ggplot(mapping = aes(x = color, y = item_size, fill = color)) + + geom_violin() + + geom_boxplot(color = "black", fill = "white", width = 0.02) + + scale_fill_brewer(palette = "Dark2", direction = -1) + + theme(legend.position = "none") + +``` + +Now that we have an idea of the relationship between the binary categories of color and the larger group of sizes, let's explore logistic regression to determine a given pumpkin's likely color. + +## 3. Build your model + +> **๐Ÿงฎ Show Me The Math** +> +> Remember how `linear regression` often used `ordinary least squares` to arrive at a value? `Logistic regression` relies on the concept of 'maximum likelihood' using [`sigmoid functions`](https://wikipedia.org/wiki/Sigmoid_function). A Sigmoid Function on a plot looks like an `S shape`. It takes a value and maps it to somewhere between 0 and 1. Its curve is also called a 'logistic curve'. Its formula looks like this: +> +> ![](../images/sigmoid.png) +> +> where the sigmoid's midpoint finds itself at x's 0 point, L is the curve's maximum value, and k is the curve's steepness. If the outcome of the function is more than 0.5, the label in question will be given the class 1 of the binary choice. If not, it will be classified as 0. + +Let's begin by splitting the data into `training` and `test` sets. The training set is used to train a classifier so that it finds a statistical relationship between the features and the label value. + +It is best practice to hold out some of your data for **testing** in order to get a better estimate of how your models will perform on new data by comparing the predicted labels with the already known labels in the test set. [rsample](https://rsample.tidymodels.org/), a package in Tidymodels, provides infrastructure for efficient data splitting and resampling: + +```{r split_data} +# Split data into 80% for training and 20% for testing +set.seed(2056) +pumpkins_split <- pumpkins_select %>% + initial_split(prop = 0.8) + +# Extract the data in each split +pumpkins_train <- training(pumpkins_split) +pumpkins_test <- testing(pumpkins_split) + +# Print out the first 5 rows of the training set +pumpkins_train %>% + slice_head(n = 5) + + +``` + +๐Ÿ™Œ We are now ready to train a model by fitting the training features to the training label (color). + +We'll begin by creating a recipe that specifies the preprocessing steps that should be carried out on our data to get it ready for modelling i.e: encoding categorical variables into a set of integers. + +There are quite a number of ways to specify a logistic regression model in Tidymodels. See `?logistic_reg()` For now, we'll specify a logistic regression model via the default `stats::glm()` engine. + +```{r log_reg} +# Create a recipe that specifies preprocessing steps for modelling +pumpkins_recipe <- recipe(color ~ ., data = pumpkins_train) %>% + step_integer(all_predictors(), zero_based = TRUE) + + +# Create a logistic model specification +log_reg <- logistic_reg() %>% + set_engine("glm") %>% + set_mode("classification") + + +``` + +Now that we have a recipe and a model specification, we need to find a way of bundling them together into an object that will first preprocess the data (prep+bake behind the scenes), fit the model on the preprocessed data and also allow for potential post-processing activities. + +In Tidymodels, this convenient object is called a [`workflow`](https://workflows.tidymodels.org/) and conveniently holds your modeling components. + +```{r workflow} +# Bundle modelling components in a workflow +log_reg_wf <- workflow() %>% + add_recipe(pumpkins_recipe) %>% + add_model(log_reg) + +# Print out the workflow +log_reg_wf + + +``` + +After a workflow has been *specified*, a model can be `trained` using the [`fit()`](https://tidymodels.github.io/parsnip/reference/fit.html) function. The workflow will estimate a recipe and preprocess the data before training, so we won't have to manually do that using prep and bake. + +```{r train} +# Train the model +wf_fit <- log_reg_wf %>% + fit(data = pumpkins_train) + +# Print the trained workflow +wf_fit + +``` + +The model print out shows the coefficients learned during training. + +Now we've trained the model using the training data, we can make predictions on the test data using [parsnip::predict()](https://parsnip.tidymodels.org/reference/predict.model_fit.html). Let's start by using the model to predict labels for our test set and the probabilities for each label. When the probability is more than 0.5, the predict class is `ORANGE` else `WHITE`. + +```{r test_pred} +# Make predictions for color and corresponding probabilities +results <- pumpkins_test %>% select(color) %>% + bind_cols(wf_fit %>% + predict(new_data = pumpkins_test)) %>% + bind_cols(wf_fit %>% + predict(new_data = pumpkins_test, type = "prob")) + +# Compare predictions +results %>% + slice_head(n = 10) + +``` + +Very nice! This provides some more insights into how logistic regression works. + +Comparing each prediction with its corresponding "ground truth" actual value isn't a very efficient way to determine how well the model is predicting. Fortunately, Tidymodels has a few more tricks up its sleeve: [`yardstick`](https://yardstick.tidymodels.org/) - a package used to measure the effectiveness of models using performance metrics. + +One performance metric associated with classification problems is the [`confusion matrix`](https://wikipedia.org/wiki/Confusion_matrix). A confusion matrix describes how well a classification model performs. A confusion matrix tabulates how many examples in each class were correctly classified by a model. In our case, it will show you how many orange pumpkins were classified as orange and how many white pumpkins were classified as white; the confusion matrix also shows you how many were classified into the **wrong** categories. + +The [**`conf_mat()`**](https://tidymodels.github.io/yardstick/reference/conf_mat.html) function from yardstick calculates this cross-tabulation of observed and predicted classes. + +```{r conf_mat} +# Confusion matrix for prediction results +conf_mat(data = results, truth = color, estimate = .pred_class) + + +``` + +Let's interpret the confusion matrix. Our model is asked to classify pumpkins between two binary categories, category `orange` and category `not-orange` + +- If your model predicts a pumpkin as orange and it belongs to category 'orange' in reality we call it a `true positive`, shown by the top left number. + +- If your model predicts a pumpkin as not orange and it belongs to category 'orange' in reality we call it a `false negative`, shown by the bottom left number. + +- If your model predicts a pumpkin as orange and it belongs to category 'not-orange' in reality we call it a `false positive`, shown by the top right number. + +- If your model predicts a pumpkin as not orange and it belongs to category 'not-orange' in reality we call it a `true negative`, shown by the bottom right number. + +| Truth | +|:-----:| + + +| | | | +|---------------|--------|-------| +| **Predicted** | ORANGE | WHITE | +| ORANGE | TP | FP | +| WHITE | FN | TN | + +As you might have guessed it's preferable to have a larger number of true positives and true negatives and a lower number of false positives and false negatives, which implies that the model performs better. + +The confusion matrix is helpful since it gives rise to other metrics that can help us better evaluate the performance of a classification model. Let's go through some of them: + +๐ŸŽ“ Precision: `TP/(TP + FP)` defined as the proportion of predicted positives that are actually positive. Also called [positive predictive value](https://en.wikipedia.org/wiki/Positive_predictive_value "Positive predictive value") + +๐ŸŽ“ Recall: `TP/(TP + FN)` defined as the proportion of positive results out of the number of samples which were actually positive. Also known as `sensitivity`. + +๐ŸŽ“ Specificity: `TN/(TN + FP)` defined as the proportion of negative results out of the number of samples which were actually negative. + +๐ŸŽ“ Accuracy: `TP + TN/(TP + TN + FP + FN)` The percentage of labels predicted accurately for a sample. + +๐ŸŽ“ F Measure: A weighted average of the precision and recall, with best being 1 and worst being 0. + +Let's calculate these metrics! + +```{r metric_set} +# Combine metric functions and calculate them all at once +eval_metrics <- metric_set(ppv, recall, spec, f_meas, accuracy) +eval_metrics(data = results, truth = color, estimate = .pred_class) +``` + +#### **Visualize the ROC curve of this model** + +For a start, this is not a bad model; its precision, recall, F measure and accuracy are in the 80% range so ideally you could use it to predict the color of a pumpkin given a set of variables. It also seems that our model was not really able to identify the white pumpkins ๐Ÿง. Could you guess why? One reason could be because of the high prevalence of ORANGE pumpkins in our training set making our model more inclined to predict the majority class. + +Let's do one more visualization to see the so-called [`ROC score`](https://en.wikipedia.org/wiki/Receiver_operating_characteristic): + +```{r roc_curve} +# Make a roc_curve +results %>% + roc_curve(color, .pred_ORANGE) %>% + autoplot() + +``` + +ROC curves are often used to get a view of the output of a classifier in terms of its true vs. false positives. ROC curves typically feature `True Positive Rate`/Sensitivity on the Y axis, and `False Positive Rate`/1-Specificity on the X axis. Thus, the steepness of the curve and the space between the midpoint line and the curve matter: you want a curve that quickly heads up and over the line. In our case, there are false positives to start with, and then the line heads up and over properly. + +Finally, let's use `yardstick::roc_auc()` to calculate the actual Area Under the Curve. One way of interpreting AUC is as the probability that the model ranks a random positive example more highly than a random negative example. + +```{r roc_aoc} +# Calculate area under curve +results %>% + roc_auc(color, .pred_ORANGE) + +``` + +The result is around `0.67053`. Given that the AUC ranges from 0 to 1, you want a big score, since a model that is 100% correct in its predictions will have an AUC of 1; in this case, the model is *pretty good*. + +In future lessons on classifications, you will learn how to improve your model's scores (such as dealing with imbalanced data in this case). + +But for now, congratulations ๐ŸŽ‰๐ŸŽ‰๐ŸŽ‰! You've completed these regression lessons! + +You R awesome! + +![Artwork by \@allison_horst](../images/r_learners_sm.jpeg) + +