{ "cells": [ { "cell_type": "markdown", "metadata": {}, "source": [ "# Logistic Regression\n", "\n", "Want to know how different inputs affects a yes/no answer? Logistic regression is your new best friend! While it's best if you've read up on linear regression first, you'll probably survive even if you didn't." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "<p class=\"reading-options\">\n <a class=\"btn\" href=\"/regression/logistic-regression\">\n <i class=\"fa fa-sm fa-book\"></i>\n Read online\n </a>\n <a class=\"btn\" href=\"/regression/notebooks/Logistic Regression.ipynb\">\n <i class=\"fa fa-sm fa-download\"></i>\n Download notebook\n </a>\n <a class=\"btn\" href=\"https://colab.research.google.com/github/littlecolumns/ds4j-notebooks/blob/master/regression/notebooks/Logistic Regression.ipynb\" target=\"_new\">\n <i class=\"fa fa-sm fa-laptop\"></i>\n Interactive version\n </a>\n</p>" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Introduction\n", "\n", "**We make scarves, and we're pretty lazy.** Since we're \ud83e\uddf6\ud83d\udc51Crafty Stylish People\ud83d\udc51\ud83e\uddf6, we might have some big questions about the scarves we knit, such as:\n", "\n", "* If we're knitting a **55-inch scarf**, how likely are we to finish it?\n", "* If we're knitting a **70-inch scarf**, how likely are we to finish it?\n", "* If we're knitting a **80-inch scarf**, how likely are we to finish it?\n", "\n", "We've been keeping records for the past few weeks, and we hope they look something like this:\n", "\n", "|\ud83d\udccf|\ud83e\udde3|\n", "|---|---|\n", "|55 inches|**Finished!**|\n", "|60 inches|**Finished!**|\n", "|65 inches|**Finished!**|\n", "|70 inches|Nope|\n", "|75 inches|Nope|\n", "|80 inches|Nope|\n", "\n", "That way we can says something like, \"if the scarf is over 65 inches, we're probably too lazy to finish it!\" Unfortunately our records do _not_ look like that. Instead, they look something like this:\n", "\n", "|\ud83d\udccf|\ud83e\udde3|\n", "|---|---|\n", "|55 inches|**Finished!**|\n", "|55 inches|**Finished!**|\n", "|55 inches|**Finished!**|\n", "|60 inches|**Finished!**|\n", "|60 inches|Nope|\n", "|70 inches|**Finished!**|\n", "|70 inches|Nope|\n", "|82 inches|**Finished!**|\n", "|82 inches|Nope|\n", "|82 inches|Nope|\n", "|82 inches|Nope|\n", "\n", "Right now our questions are about our chances of finishing a 55-inch, 70-inch and 80-inch scarf. We can _try_ to answer some of our questions with this data: for 55-inch scarves we've **finished all of them**, and we've finished **half of the 70-inch scarves**. There's a problem investigating our chances of finishing an 80-inch scarf, though: **we don't have any 80-inch scarves in our dataset!** \n", "\n", "To reason about our chances of finishing an 80-inch scarf, let's think about what we do know:\n", "\n", "* We rarely finish 82-inch scarves (a little longer than 80 inches)\n", "* We finish about half of the 70-inch and 60-inch scarves (a good amount shorter)\n", "* We're very good at finishing 55-inch scarves (much much shorter)\n", "\n", "It sounds like we're **generally bad at finishing longer scarves,** which is totally understandable. But _how_ bad?\n", "\n", "**This is where logistic regression rescues us.** Logistic regression can tell us two things:\n", "\n", "* What the effect of length is on our ability to complete a scarf\n", "* Given a length, what are our chances of completing it?\n", "\n", "Completing our scarf is a yes/no question, right? **Logistic regression is all about predicting categories** - in this case, the category yes vs the category no. If you're predicting a number instead, you use linear regression. That's it!\n", "\n", "Let's see what this looks like with a little code.\n", "\n", "> For the questioning souls: Yes, the \"chances of finishing\" are a number, but the inputs are yes/no or completed/incomplete. That makes it a logistic regression." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Performing a logistic regression\n", "\n", "We'll start off with our data. We're going to use [pandas](https://pandas.pydata.org/), a super-popular Python library for doing data-y things." ] }, { "cell_type": "code", "execution_count": 1, "metadata": {}, "outputs": [ { "data": { "text/html": [ "<div>\n", "<style scoped>\n", " .dataframe tbody tr th:only-of-type {\n", " vertical-align: middle;\n", " }\n", "\n", " .dataframe tbody tr th {\n", " vertical-align: top;\n", " }\n", "\n", " .dataframe thead th {\n", " text-align: right;\n", " }\n", "</style>\n", "<table border=\"1\" class=\"dataframe\">\n", " <thead>\n", " <tr style=\"text-align: right;\">\n", " <th></th>\n", " <th>length_in</th>\n", " <th>completed</th>\n", " </tr>\n", " </thead>\n", " <tbody>\n", " <tr>\n", " <th>0</th>\n", " <td>55</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>1</th>\n", " <td>55</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>2</th>\n", " <td>55</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>3</th>\n", " <td>60</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>4</th>\n", " <td>60</td>\n", " <td>0</td>\n", " </tr>\n", " <tr>\n", " <th>5</th>\n", " <td>70</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>6</th>\n", " <td>70</td>\n", " <td>0</td>\n", " </tr>\n", " <tr>\n", " <th>7</th>\n", " <td>82</td>\n", " <td>1</td>\n", " </tr>\n", " <tr>\n", " <th>8</th>\n", " <td>82</td>\n", " <td>0</td>\n", " </tr>\n", " <tr>\n", " <th>9</th>\n", " <td>82</td>\n", " <td>0</td>\n", " </tr>\n", " <tr>\n", " <th>10</th>\n", " <td>82</td>\n", " <td>0</td>\n", " </tr>\n", " </tbody>\n", "</table>\n", "</div>" ], "text/plain": [ " length_in completed\n", "0 55 1\n", "1 55 1\n", "2 55 1\n", "3 60 1\n", "4 60 0\n", "5 70 1\n", "6 70 0\n", "7 82 1\n", "8 82 0\n", "9 82 0\n", "10 82 0" ] }, "execution_count": 1, "metadata": {}, "output_type": "execute_result" } ], "source": [ "import pandas as pd\n", "import numpy as np\n", "\n", "df = pd.DataFrame([\n", " { 'length_in': 55, 'completed': 1 },\n", " { 'length_in': 55, 'completed': 1 },\n", " { 'length_in': 55, 'completed': 1 },\n", " { 'length_in': 60, 'completed': 1 },\n", " { 'length_in': 60, 'completed': 0 },\n", " { 'length_in': 70, 'completed': 1 },\n", " { 'length_in': 70, 'completed': 0 },\n", " { 'length_in': 82, 'completed': 1 },\n", " { 'length_in': 82, 'completed': 0 },\n", " { 'length_in': 82, 'completed': 0 },\n", " { 'length_in': 82, 'completed': 0 },\n", "])\n", "df" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Our very tiny dataset has two columns:\n", "\n", "* Number of inches in the atttempted scarf\n", "* Whether we completed it or not. 1 means yes, 0 means no\n", "\n", "We want to ask a simple question using this data: **how does length affect our ability to finish a scarf?** Since finishing a scarf is a yes/no question, we get to use **logistic regression**.\n", "\n", "### Writing a regression\n", "\n", "To perform our logistic regression, we're going to use a library called [statsmodels](https://www.statsmodels.org), the same one we used for linear regression.\n", "\n", "We'll use `smf.logit` to create a new logistic regression, then use `model.fit()` to teach it all about scarf-making." ] }, { "cell_type": "code", "execution_count": 2, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "Optimization terminated successfully.\n", " Current function value: 0.531806\n", " Iterations 5\n" ] } ], "source": [ "import statsmodels.formula.api as smf\n", "\n", "# What effect does the length of the scarf have one whether it was completed?\n", "model = smf.logit(formula='completed ~ length_in', data=df)\n", "results = model.fit()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The technique of writing out the relationship like `\"completed ~ length_in\"` was stolen from R. It uses a library called [Patsy](https://patsy.readthedocs.io/), and you can read all about it in [the statsmodels documentation](https://www.statsmodels.org/stable/example_formulas.html).\n", "\n", "> When using statsmodels, linear regression is `smf.ols` while logistic is `smf.logit`." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Making predictions\n", "\n", "Now that we've taught our regression all about how bad we are at making scarves, let's **make some predictions** to see what it learned. Along with the 80-inch scarf, let's try out a few more sizes as well." ] }, { "cell_type": "code", "execution_count": 3, "metadata": {}, "outputs": [ { "data": { "text/html": [ "<div>\n", "<style scoped>\n", " .dataframe tbody tr th:only-of-type {\n", " vertical-align: middle;\n", " }\n", "\n", " .dataframe tbody tr th {\n", " vertical-align: top;\n", " }\n", "\n", " .dataframe thead th {\n", " text-align: right;\n", " }\n", "</style>\n", "<table border=\"1\" class=\"dataframe\">\n", " <thead>\n", " <tr style=\"text-align: right;\">\n", " <th></th>\n", " <th>length_in</th>\n", " <th>predicted</th>\n", " </tr>\n", " </thead>\n", " <tbody>\n", " <tr>\n", " <th>0</th>\n", " <td>55</td>\n", " <td>0.850526</td>\n", " </tr>\n", " <tr>\n", " <th>1</th>\n", " <td>65</td>\n", " <td>0.651815</td>\n", " </tr>\n", " <tr>\n", " <th>2</th>\n", " <td>75</td>\n", " <td>0.381148</td>\n", " </tr>\n", " <tr>\n", " <th>3</th>\n", " <td>80</td>\n", " <td>0.261047</td>\n", " </tr>\n", " <tr>\n", " <th>4</th>\n", " <td>90</td>\n", " <td>0.104122</td>\n", " </tr>\n", " </tbody>\n", "</table>\n", "</div>" ], "text/plain": [ " length_in predicted\n", "0 55 0.850526\n", "1 65 0.651815\n", "2 75 0.381148\n", "3 80 0.261047\n", "4 90 0.104122" ] }, "execution_count": 3, "metadata": {}, "output_type": "execute_result" } ], "source": [ "# What lengths do we want to ask it about?\n", "unknown = pd.DataFrame([\n", " { 'length_in': 55 },\n", " { 'length_in': 65 },\n", " { 'length_in': 75 },\n", " { 'length_in': 80 },\n", " { 'length_in': 90 },\n", "])\n", "\n", "# Save the predictions into a new column\n", "unknown['predicted'] = results.predict(unknown)\n", "unknown" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "According to the regression, we have a **26% chance to finish a 80-inch scarf**. While I'd like to say we can do better.... like I said, we're lazy.\n", "\n", "On the other hand, we only have a 85% chance for our 55-inch scarves!! Even though we finished *every single 55-inch scarf we've ever made*, it turns out that not completing a handful of 60-, 70-, and 82-inch scarves has made our regression a little pessimistic." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Examining our results\n", "\n", "Usually when you're doing logistic regression you don't care too much about making predictions. You're more concerned with figuring out things like how good your regression was, and - in this case - **how much each inch adds to our chance of failure**.\n", "\n", "There are a few ways to look at the results, but we'll only use the fanciest since we like \u2728\ud83c\udf1f\ud83d\udc8e \ud835\udcbb\ud835\udcb6\ud835\udcc3\ud835\udcb8\ud835\udcce \ud835\udcc9\ud835\udcbd\ud835\udcbe\ud835\udcc3\ud835\udc54\ud835\udcc8 \ud83d\udc8e\ud83c\udf1f\u2728. To get the results we just look at `results.summary()`:" ] }, { "cell_type": "code", "execution_count": 4, "metadata": {}, "outputs": [ { "data": { "text/html": [ "<table class=\"simpletable\">\n", "<caption>Logit Regression Results</caption>\n", "<tr>\n", " <th>Dep. Variable:</th> <td>completed</td> <th> No. Observations: </th> <td> 11</td> \n", "</tr>\n", "<tr>\n", " <th>Model:</th> <td>Logit</td> <th> Df Residuals: </th> <td> 9</td> \n", "</tr>\n", "<tr>\n", " <th>Method:</th> <td>MLE</td> <th> Df Model: </th> <td> 1</td> \n", "</tr>\n", "<tr>\n", " <th>Date:</th> <td>Mon, 30 Dec 2019</td> <th> Pseudo R-squ.: </th> <td>0.2282</td> \n", "</tr>\n", "<tr>\n", " <th>Time:</th> <td>12:07:56</td> <th> Log-Likelihood: </th> <td> -5.8499</td>\n", "</tr>\n", "<tr>\n", " <th>converged:</th> <td>True</td> <th> LL-Null: </th> <td> -7.5791</td>\n", "</tr>\n", "<tr>\n", " <th>Covariance Type:</th> <td>nonrobust</td> <th> LLR p-value: </th> <td>0.06293</td>\n", "</tr>\n", "</table>\n", "<table class=\"simpletable\">\n", "<tr>\n", " <td></td> <th>coef</th> <th>std err</th> <th>z</th> <th>P>|z|</th> <th>[0.025</th> <th>0.975]</th> \n", "</tr>\n", "<tr>\n", " <th>Intercept</th> <td> 7.8531</td> <td> 4.736</td> <td> 1.658</td> <td> 0.097</td> <td> -1.429</td> <td> 17.135</td>\n", "</tr>\n", "<tr>\n", " <th>length_in</th> <td> -0.1112</td> <td> 0.067</td> <td> -1.649</td> <td> 0.099</td> <td> -0.243</td> <td> 0.021</td>\n", "</tr>\n", "</table>" ], "text/plain": [ "<class 'statsmodels.iolib.summary.Summary'>\n", "\"\"\"\n", " Logit Regression Results \n", "==============================================================================\n", "Dep. Variable: completed No. Observations: 11\n", "Model: Logit Df Residuals: 9\n", "Method: MLE Df Model: 1\n", "Date: Mon, 30 Dec 2019 Pseudo R-squ.: 0.2282\n", "Time: 12:07:56 Log-Likelihood: -5.8499\n", "converged: True LL-Null: -7.5791\n", "Covariance Type: nonrobust LLR p-value: 0.06293\n", "==============================================================================\n", " coef std err z P>|z| [0.025 0.975]\n", "------------------------------------------------------------------------------\n", "Intercept 7.8531 4.736 1.658 0.097 -1.429 17.135\n", "length_in -0.1112 0.067 -1.649 0.099 -0.243 0.021\n", "==============================================================================\n", "\"\"\"" ] }, "execution_count": 4, "metadata": {}, "output_type": "execute_result" } ], "source": [ "results.summary()" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The part we're interested in is down at the bottom, where it says `length_in` and `coef`. `coef` stands for **coefficient**, and it's (kind of) the \"how much\" in \"how much does length affect completion.\"\n", "\n", "<img src=\"\">\n", "\n", "In this case, `length_in` has a coefficient of around **-0.1**. With linear regression, we could use the coefficient to put things into words easily, saying something like \"every 1000 miles we get in one more accident.\" Sadly, **it's not so easy with logistic regression!**\n", "\n", "## Odds and odds ratios\n", "\n", "LOGistic regression is, yes, all about LOGarithms, and to turn this coefficient into something cool and normal we need to ask Python to undo the logarithm. The opposite of a logarithm is an exponent, so we're going to ask [np.exp](https://docs.scipy.org/doc/numpy/reference/generated/numpy.exp.html) to perform a little magic for us." ] }, { "cell_type": "code", "execution_count": 5, "metadata": {}, "outputs": [ { "data": { "text/html": [ "<div>\n", "<style scoped>\n", " .dataframe tbody tr th:only-of-type {\n", " vertical-align: middle;\n", " }\n", "\n", " .dataframe tbody tr th {\n", " vertical-align: top;\n", " }\n", "\n", " .dataframe thead th {\n", " text-align: right;\n", " }\n", "</style>\n", "<table border=\"1\" class=\"dataframe\">\n", " <thead>\n", " <tr style=\"text-align: right;\">\n", " <th></th>\n", " <th>coef</th>\n", " <th>odds ratio</th>\n", " <th>name</th>\n", " </tr>\n", " </thead>\n", " <tbody>\n", " <tr>\n", " <th>0</th>\n", " <td>7.853131</td>\n", " <td>2573.780516</td>\n", " <td>Intercept</td>\n", " </tr>\n", " <tr>\n", " <th>1</th>\n", " <td>-0.111171</td>\n", " <td>0.894786</td>\n", " <td>length_in</td>\n", " </tr>\n", " </tbody>\n", "</table>\n", "</div>" ], "text/plain": [ " coef odds ratio name\n", "0 7.853131 2573.780516 Intercept\n", "1 -0.111171 0.894786 length_in" ] }, "execution_count": 5, "metadata": {}, "output_type": "execute_result" } ], "source": [ "coefs = pd.DataFrame({\n", " 'coef': results.params.values,\n", " 'odds ratio': np.exp(results.params.values),\n", " 'name': results.params.index\n", "})\n", "coefs" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "The **odds ratio** column tells you what happens to the **odds** of completing a scarf each time you add one more of each variable.\n", "\n", "* If the odds ratio is 1, it means nothing changes as we add or remove inches.\n", "* If it's above 1, the odds increase. For example, odds ratio of 2.0 would mean our odds to finish are twice as good for each additional inch.\n", "* If it's below 1, the odds decrease. For example, an odds ratio of 0.5 would mean our odds drop by 0.5x (half) for each additional inch.\n", "\n", "In this case, `length_in` has an odds ratio of about 0.89. This means for each extra inch we add, we have 0.89x the odds we did before - a drop of 11%.\n", "\n", "## What are odds?\n", "\n", "Even though we keep talking about odds, let's be honest: the worst thing in history at the moment is that **you probably have no idea what odds are**. You're _definitely_ (probably maybe) thinking about **probability instead**, because in real life we never ever ever talk about odds.\n", "\n", "Odds and probability (or % chance) are closely related to each other, though. Let's compare the two:\n", "\n", "|words|probability|odds|\n", "|---|---|---|\n", "|---|`finished / attempted`|`finished / unfinished`|\n", "|I start on 100 scarves, I finish half|50/100 = **0.5 or 50% chance**|50:50 odds, 50/50 = **1.0 or 1:1 odds**|\n", "|I finish 20, ignore 80|20/100 = **0.2 or 20% chance**|20:80 odds, 20/80 = **0.25 or 1:4 odds**|\n", "|I finish 75, ignore 25|75/100 = **0.75 or 75% chance**|75:25 odds, 75/25 = **3.0 or 3:1 odds**|\n", "|I finish 996, ignore 4|996/1000 = **0.996 or 99.6% chance**|99.6:0.4 odds, 99.6/0.4 = **249.0 or 249:1 odds**|\n", "\n", "Yeah, it gets pretty weird. If you're at a 100% chance, the odds are 100:0, or 100/0, which is _infinity_.\n", "\n", "### Odds and probability conversion\n", "\n", "To try and make this up to you, I'll show you how to escape from odds and back to probability (...is that even a reward?). Here are a couple formulas to convert between the two measurements:\n", "\n", "```python\n", "probability = odds / (odds + 1)\n", "odds = probability / (1 - probability)\n", "```\n", "\n", "For example, up above our **probability** for finishing a 55-inch scarf was about **0.85**, or 85%. What is that probability as odds?" ] }, { "cell_type": "code", "execution_count": 6, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "5.666666666666666" ] }, "execution_count": 6, "metadata": {}, "output_type": "execute_result" } ], "source": [ "0.85 / (1 - 0.85)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "According to The Power Of Mathematics, an 85% probability is an odds of around 5.67." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Using the odds ratio\n", "\n", "There's a reason we're spending so much time discussing this, and it's so we can talk about the **odds ratio**. Remember when we looked at the coefficient/odds ratio table before?" ] }, { "cell_type": "code", "execution_count": 7, "metadata": {}, "outputs": [ { "data": { "text/html": [ "<div>\n", "<style scoped>\n", " .dataframe tbody tr th:only-of-type {\n", " vertical-align: middle;\n", " }\n", "\n", " .dataframe tbody tr th {\n", " vertical-align: top;\n", " }\n", "\n", " .dataframe thead th {\n", " text-align: right;\n", " }\n", "</style>\n", "<table border=\"1\" class=\"dataframe\">\n", " <thead>\n", " <tr style=\"text-align: right;\">\n", " <th></th>\n", " <th>coef</th>\n", " <th>odds ratio</th>\n", " <th>name</th>\n", " </tr>\n", " </thead>\n", " <tbody>\n", " <tr>\n", " <th>0</th>\n", " <td>7.853131</td>\n", " <td>2573.780516</td>\n", " <td>Intercept</td>\n", " </tr>\n", " <tr>\n", " <th>1</th>\n", " <td>-0.111171</td>\n", " <td>0.894786</td>\n", " <td>length_in</td>\n", " </tr>\n", " </tbody>\n", "</table>\n", "</div>" ], "text/plain": [ " coef odds ratio name\n", "0 7.853131 2573.780516 Intercept\n", "1 -0.111171 0.894786 length_in" ] }, "execution_count": 7, "metadata": {}, "output_type": "execute_result" } ], "source": [ "coefs = pd.DataFrame({\n", " 'coef': results.params.values,\n", " 'odds ratio': np.exp(results.params.values),\n", " 'name': results.params.index\n", "})\n", "coefs" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "As we saw above:\n", "\n", "> In this case, `length_in` has an **odds ratio** of about 0.89. This means for each extra inch we add, we have 0.89x the odds we did before (aka 11% less).\n", "\n", "The odds ratio is important in data-driven reporting because it's often used to **explore charges of bias**. For example, you might find that being African American increases the odds of being searched at a traffic stop, or living in a low-income neighborhood decreases your odds of a mortgage being approved. **Learning to use the odds ratio helps you successfully investigate and report on these topics.**\n", "\n", "Time to put our odds ratio to work!\n", "\n", "Let's say we're aiming for a 55-inch scarf, but we'd maybe like it a little longer. But if we're less likely to finish it if it's longer, maybe we should keep it short? Let's use our odds ratio to see how our chance of finishing a 55-inch scarf changes as we **add inches**.\n", "\n", "To start, we need the **odds** for completing a 55-inch scarf. When we did the prediction up above, it gave us a **probability of 0.85** for a 55-inch scarf. Let's convert that probability to odds using our `probability / (1 - probability)` formula." ] }, { "cell_type": "code", "execution_count": 8, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "5.666666666666666" ] }, "execution_count": 8, "metadata": {}, "output_type": "execute_result" } ], "source": [ "# Odds for finishing a 55-inch scarf\n", "odds_55 = 0.85 / (1 - 0.85)\n", "odds_55" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "If we want to add an inch, we use the coefficient/odds ratio table above to find the **odds ratio** of 0.895. If we go one inch longer, we have to multiply the odds by the odds ratio, 0.895. Note that since the odds ratio is **less than 1**, that will make the odds go **down**." ] }, { "cell_type": "code", "execution_count": 9, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "5.071666666666666" ] }, "execution_count": 9, "metadata": {}, "output_type": "execute_result" } ], "source": [ "# Odds for a 56 inch scarf\n", "# Multiply the 55-inch scarf odds by the odds ratio\n", "odds_55 * 0.895" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "We'll multiple it twice if we want to add two inches, moving up to a 57 inch scarf." ] }, { "cell_type": "code", "execution_count": 10, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "4.539141666666667" ] }, "execution_count": 10, "metadata": {}, "output_type": "execute_result" } ], "source": [ "# Odds for a 57 inch scarf\n", "# Multiply the 55-inch scarf odds by the odds ratio twice\n", "odds_55 * 0.895 * 0.895" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "And one more for a 58-inch scarf..." ] }, { "cell_type": "code", "execution_count": 11, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "4.062531791666666" ] }, "execution_count": 11, "metadata": {}, "output_type": "execute_result" } ], "source": [ "# Odds for a 58 inch scarf\n", "odds_55 * 0.895 * 0.895 * 0.895" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "And now we'll jump ahead to a 60-inch scarf. Five extra inches means five times multiplied by the odds ratio." ] }, { "cell_type": "code", "execution_count": 12, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "3.2541895284197917" ] }, "execution_count": 12, "metadata": {}, "output_type": "execute_result" } ], "source": [ "odds_55 * 0.895 * 0.895 * 0.895 * 0.895 * 0.895" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "Even though we did all this math, we're still not really hyped about a 3.25 odds ratio. **No one understands odds, what is this number in probability?** We can use the `probability = odds / (odds + 1)` formula." ] }, { "cell_type": "code", "execution_count": 13, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "0.7649376001422655" ] }, "execution_count": 13, "metadata": {}, "output_type": "execute_result" } ], "source": [ "odds_60 = odds_55 * 0.895 * 0.895 * 0.895 * 0.895 * 0.895\n", "odds_60 / (odds_60 + 1)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "There we go! Adding five extra inches dropped our probability of completion from 85% to 76%.\n", "\n", "We calculated this the long long confusing way, though. Remember our friend `results.predict`? It allows us to **make predictions** if we give it a dataframe. Let's feed it the details for a sixty-inch scarf and see what it thinks the probability should be." ] }, { "cell_type": "code", "execution_count": 14, "metadata": {}, "outputs": [ { "data": { "text/plain": [ "0 0.765466\n", "dtype: float64" ] }, "execution_count": 14, "metadata": {}, "output_type": "execute_result" } ], "source": [ "sample_df = pd.DataFrame([\n", " {'length_in': 60}\n", "])\n", "\n", "results.predict(sample_df)" ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "**It matches our calculation perfectly!** ...or close enough, since we rounded 0.894786 to 0.895 so we could save a little bit of typing." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "### Why not percent change?\n", "\n", "**Why don't we just talk about percent increase instead of odds ratio?** Even though that would make way way way more sense to us, we can't! Let's look at why.\n", "\n", "Say we have a really skilled friend who helps us out with knitting, and her help has an **odds ratio of 2.0**. She doubles the odds of completion!!! Incredible!!! But because **odds ratios are about odds and _not_ probability**, her help does **not** take us from 25% to 50%. Here's what a 2x increase in odds does:\n", "\n", "|probability|odds|odds are doubled|convert to %|new percentage|increase|\n", "|---|---|---|---|---|---|\n", "|50/100 or 50%|1:1 or **1.0**|2|`2 / (2 + 1)`|0.67 or **67%**|+17 percentage points|\n", "|20/100 or 20%|1:4 or **0.25**|0.5| `0.5 / (0.5 + 1)`|0.33 or **33%**|+13 percentage points|\n", "|75/100 or 75%|3:1 or **3.0**|6| `6 / (6 + 1)`|0.86 or **86%**|+11 percentage points|\n", "\n", "Even though it was all a 2x odds ratio, each one had a different percent increase! **An odds ratio above 1.0 will always increase your probability, but the change depends on the probability you started at.** For example, the probability change from 55 inches to 57 inches will be different than than 65 inches to 67 inches, even though they're both a 2-inch increase." ] }, { "cell_type": "markdown", "metadata": {}, "source": [ "## Review\n", "\n", "TODO" ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [] } ], "metadata": { "kernelspec": { "display_name": "Python 3", "language": "python", "name": "python3" }, "language_info": { "codemirror_mode": { "name": "ipython", "version": 3 }, "file_extension": ".py", "mimetype": "text/x-python", "name": "python", "nbconvert_exporter": "python", "pygments_lexer": "ipython3", "version": "3.6.8" }, "toc": { "base_numbering": 1, "nav_menu": {}, "number_sections": true, "sideBar": true, "skip_h1_title": false, "title_cell": "Table of Contents", "title_sidebar": "Contents", "toc_cell": false, "toc_position": {}, "toc_section_display": true, "toc_window_display": false } }, "nbformat": 4, "nbformat_minor": 2 }