{ "nbformat": 4, "nbformat_minor": 0, "metadata": { "colab": { "name": "lesson_1-R.ipynb", "provenance": [], "collapsed_sections": [], "toc_visible": true }, "kernelspec": { "name": "ir", "display_name": "R" }, "language_info": { "name": "R" } }, "cells": [ { "cell_type": "markdown", "metadata": { "id": "YJUHCXqK57yz" }, "source": [ "#Build a regression model: Get started with R and Tidymodels for regression models" ] }, { "cell_type": "markdown", "metadata": { "id": "LWNNzfqd6feZ" }, "source": [ "## Introduction to Regression - Lesson 1\n", "\n", "#### Putting it into perspective\n", "\n", "✅ There are many types of regression methods, and which one you pick depends on the answer you're looking for. If you want to predict the probable height for a person of a given age, you'd use `linear regression`, as you're seeking a **numeric value**. If you're interested in discovering whether a type of cuisine should be considered vegan or not, you're looking for a **category assignment** so you would use `logistic regression`. You'll learn more about logistic regression later. Think a bit about some questions you can ask of data, and which of these methods would be more appropriate.\n", "\n", "In this section, you will work with a [small dataset about diabetes](https://www4.stat.ncsu.edu/~boos/var.select/diabetes.html). Imagine that you wanted to test a treatment for diabetic patients. Machine Learning models might help you determine which patients would respond better to the treatment, based on combinations of variables. Even a very basic regression model, when visualized, might show information about variables that would help you organize your theoretical clinical trials.\n", "\n", "That said, let's get started on this task!\n", "\n", "![Artwork by \\@allison_horst](../images/encouRage.jpg)
Artwork by @allison_horst" ] }, { "cell_type": "markdown", "metadata": { "id": "FIo2YhO26wI9" }, "source": [ "## 1. Loading up our tool set\n", "\n", "For this task, 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", "You can have them installed as:\n", "\n", "`install.packages(c(\"tidyverse\", \"tidymodels\"))`\n", "\n", "The script below checks whether you have the packages required to complete this module and installs them for you in case some are missing." ] }, { "cell_type": "code", "metadata": { "id": "cIA9fz9v7Dss", "colab": { "base_uri": "https://localhost:8080/" }, "outputId": "2df7073b-86b2-4b32-cb86-0da605a0dc11" }, "source": [ "if (!require(\"pacman\")) install.packages(\"pacman\")\n", "pacman::p_load(tidyverse, tidymodels)" ], "execution_count": 2, "outputs": [ { "output_type": "stream", "text": [ "Loading required package: pacman\n", "\n" ], "name": "stderr" } ] }, { "cell_type": "markdown", "metadata": { "id": "gpO_P_6f9WUG" }, "source": [ "Now, let's load these awesome packages and make them available in our current R session.(This is for mere illustration, `pacman::p_load()` already did that for you)" ] }, { "cell_type": "code", "metadata": { "id": "NLMycgG-9ezO" }, "source": [ "# load the core Tidyverse packages\n", "library(tidyverse)\n", "\n", "# load the core Tidymodels packages\n", "library(tidymodels)\n" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "KM6iXLH996Cl" }, "source": [ "## 2. The diabetes dataset\n", "\n", "In this exercise, we'll put our regression skills into display by making predictions on a diabetes dataset. The [diabetes dataset](https://www4.stat.ncsu.edu/~boos/var.select/diabetes.rwrite1.txt) includes `442 samples` of data around diabetes, with 10 predictor feature variables, `age`, `sex`, `body mass index`, `average blood pressure`, and `six blood serum measurements` as well as an outcome variable `y`: a quantitative measure of disease progression one year after baseline.\n", "\n", "|Number of observations|442|\n", "|----------------------|:---|\n", "|Number of predictors|First 10 columns are numeric predictive|\n", "|Outcome/Target|Column 11 is a quantitative measure of disease progression one year after baseline|\n", "|Predictor Information|- age in years\n", "||- sex\n", "||- bmi body mass index\n", "||- bp average blood pressure\n", "||- s1 tc, total serum cholesterol\n", "||- s2 ldl, low-density lipoproteins\n", "||- s3 hdl, high-density lipoproteins\n", "||- s4 tch, total cholesterol / HDL\n", "||- s5 ltg, possibly log of serum triglycerides level\n", "||- s6 glu, blood sugar level|\n", "\n", "\n", "\n", "\n", "> 🎓 Remember, this is supervised learning, and we need a named 'y' target.\n", "\n", "Before you can manipulate data with R, you need to import the data into R's memory, or build a connection to the data that R can use to access the data remotely.\n", "\n", "> The [readr](https://readr.tidyverse.org/) package, which is part of the Tidyverse, provides a fast and friendly way to read rectangular data into R.\n", "\n", "Now, let's load the diabetes dataset provided in this source URL: \n", "\n", "Also, we'll perform a sanity check on our data using `glimpse()` and dsiplay the first 5 rows using `slice()`.\n", "\n", "Before going any further, let's also introduce something you will encounter often in R code 🥁🥁: the pipe operator `%>%`\n", "\n", "The pipe operator (`%>%`) performs operations in logical sequence by passing an object forward into a function or call expression. You can think of the pipe operator as saying \"and then\" in your code." ] }, { "cell_type": "code", "metadata": { "id": "Z1geAMhM-bSP" }, "source": [ "# Import the data set\n", "diabetes <- read_table2(file = \"https://www4.stat.ncsu.edu/~boos/var.select/diabetes.rwrite1.txt\")\n", "\n", "\n", "# Get a glimpse and dimensions of the data\n", "glimpse(diabetes)\n", "\n", "\n", "# Select the first 5 rows of the data\n", "diabetes %>% \n", " slice(1:5)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "UwjVT1Hz-c3Z" }, "source": [ "`glimpse()` shows us that this data has 442 rows and 11 columns with all the columns being of data type `double` \n", "\n", "
\n", "\n", "\n", "\n", "> glimpse() and slice() are functions in [`dplyr`](https://dplyr.tidyverse.org/). Dplyr, part of the Tidyverse, is a grammar of data manipulation that provides a consistent set of verbs that help you solve the most common data manipulation challenges\n", "\n", "
\n", "\n", "Now that we have the data, let's narrow down to one feature (`bmi`) to target for this exercise. This will require us to select the desired columns. So, how do we do this?\n", "\n", "[`dplyr::select()`](https://dplyr.tidyverse.org/reference/select.html) allows us to *select* (and optionally rename) columns in a data frame." ] }, { "cell_type": "code", "metadata": { "id": "RDY1oAKI-m80" }, "source": [ "# Select predictor feature `bmi` and outcome `y`\n", "diabetes_select <- diabetes %>% \n", " select(c(bmi, y))\n", "\n", "# Print the first 5 rows\n", "diabetes_select %>% \n", " slice(1:10)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "SDk668xK-tc3" }, "source": [ "## 3. Training and Testing data\n", "\n", "It's common practice in supervised learning to *split* the data into two subsets; a (typically larger) set with which to train the model, and a smaller \"hold-back\" set with which to see how the model performed.\n", "\n", "Now that we have data ready, we can see if a machine can help determine a logical split between the numbers in this dataset. We can use the [rsample](https://tidymodels.github.io/rsample/) package, which is part of the Tidymodels framework, to create an object that contains the information on *how* to split the data, and then two more rsample functions to extract the created training and testing sets:\n" ] }, { "cell_type": "code", "metadata": { "id": "EqtHx129-1h-" }, "source": [ "set.seed(2056)\n", "# Split 67% of the data for training and the rest for tesing\n", "diabetes_split <- diabetes_select %>% \n", " initial_split(prop = 0.67)\n", "\n", "# Extract the resulting train and test sets\n", "diabetes_train <- training(diabetes_split)\n", "diabetes_test <- testing(diabetes_split)\n", "\n", "# Print the first 3 rows of the training set\n", "diabetes_train %>% \n", " slice(1:10)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "sBOS-XhB-6v7" }, "source": [ "## 4. Train a linear regression model with Tidymodels\n", "\n", "Now we are ready to train our model!\n", "\n", "In Tidymodels, you specify models using `parsnip()` by specifying three concepts:\n", "\n", "- Model **type** differentiates models such as linear regression, logistic regression, decision tree models, and so forth.\n", "\n", "- Model **mode** includes common options like regression and classification; some model types support either of these while some only have one mode.\n", "\n", "- Model **engine** is the computational tool which will be used to fit the model. Often these are R packages, such as **`\"lm\"`** or **`\"ranger\"`**\n", "\n", "This modeling information is captured in a model specification, so let's build one!" ] }, { "cell_type": "code", "metadata": { "id": "20OwEw20--t3" }, "source": [ "# Build a linear model specification\n", "lm_spec <- \n", " # Type\n", " linear_reg() %>% \n", " # Engine\n", " set_engine(\"lm\") %>% \n", " # Mode\n", " set_mode(\"regression\")\n", "\n", "\n", "# Print the model specification\n", "lm_spec" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "_oDHs89k_CJj" }, "source": [ "After a model has been *specified*, the model can be `estimated` or `trained` using the [`fit()`](https://parsnip.tidymodels.org/reference/fit.html) function, typically using a formula and some data.\n", "\n", "`y ~ .` means we'll fit `y` as the predicted quantity/target, explained by all the predictors/features ie, `.` (in this case, we only have one predictor: `bmi` )" ] }, { "cell_type": "code", "metadata": { "id": "YlsHqd-q_GJQ" }, "source": [ "# Build a linear model specification\n", "lm_spec <- linear_reg() %>% \n", " set_engine(\"lm\") %>%\n", " set_mode(\"regression\")\n", "\n", "\n", "# Train a linear regression model\n", "lm_mod <- lm_spec %>% \n", " fit(y ~ ., data = diabetes_train)\n", "\n", "# Print the model\n", "lm_mod" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "kGZ22RQj_Olu" }, "source": [ "From the model output, we can see the coefficients learned during training. They represent the coefficients of the line of best fit that gives us the lowest overall error between the actual and predicted variable.\n", "
\n", "\n", "## 5. Make predictions on the test set\n", "\n", "Now that we've trained a model, we can use it to predict the disease progression y for the test dataset using [parsnip::predict()](https://parsnip.tidymodels.org/reference/predict.model_fit.html). This will be used to draw the line between data groups." ] }, { "cell_type": "code", "metadata": { "id": "nXHbY7M2_aao" }, "source": [ "# Make predictions for the test set\n", "predictions <- lm_mod %>% \n", " predict(new_data = diabetes_test)\n", "\n", "# Print out some of the predictions\n", "predictions %>% \n", " slice(1:5)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "R_JstwUY_bIs" }, "source": [ "Woohoo! 💃🕺 We just trained a model and used it to make predictions!\n", "\n", "When making predictions, the tidymodels convention is to always produce a tibble/data frame of results with standardized column names. This makes it easy to combine the original data and the predictions in a usable format for subsequent operations such as plotting.\n", "\n", "`dplyr::bind_cols()` efficiently binds multiple data frames column." ] }, { "cell_type": "code", "metadata": { "id": "RybsMJR7_iI8" }, "source": [ "# Combine the predictions and the original test set\n", "results <- diabetes_test %>% \n", " bind_cols(predictions)\n", "\n", "\n", "results %>% \n", " slice(1:5)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "XJbYbMZW_n_s" }, "source": [ "## 6. Plot modelling results\n", "\n", "Now, its time to see this visually 📈. We'll create a scatter plot of all the `y` and `bmi` values of the test set, then use the predictions to draw a line in the most appropriate place, between the model's data groupings.\n", "\n", "R has several systems for making graphs, but `ggplot2` is one of the most elegant and most versatile. This allows you to compose graphs by **combining independent components**." ] }, { "cell_type": "code", "metadata": { "id": "R9tYp3VW_sTn" }, "source": [ "# Set a theme for the plot\n", "theme_set(theme_light())\n", "# Create a scatter plot\n", "results %>% \n", " ggplot(aes(x = bmi)) +\n", " # Add a scatter plot\n", " geom_point(aes(y = y), size = 1.6) +\n", " # Add a line plot\n", " geom_line(aes(y = .pred), color = \"blue\", size = 1.5)" ], "execution_count": null, "outputs": [] }, { "cell_type": "markdown", "metadata": { "id": "zrPtHIxx_tNI" }, "source": [ "> ✅ Think a bit about what's going on here. A straight line is running through many small dots of data, but what is it doing exactly? Can you see how you should be able to use this line to predict where a new, unseen data point should fit in relationship to the plot's y axis? Try to put into words the practical use of this model.\n", "\n", "Congratulations, you built your first linear regression model, created a prediction with it, and displayed it in a plot!\n" ] } ] }