Groups
Plotting Distributions over Categories
Slides and Tutorials
R (Static Viz) | Radiant Tutorial | Datasets |
“In keeping silent about evil, in burying it so deep within us that no sign of it appears on the surface, we are implanting it, and it will rise up a thousand fold in the future.”
— Aleksandr Solzhenitsyn
Setting up R Packages
What graphs will we see today?
Variable #1 | Variable #2 | Chart Names | Chart Shape | |
---|---|---|---|---|
Quant | Qual | Box Plot |
What kind of Data Variables will we choose?
No | Pronoun | Answer | Variable/Scale | Example | What Operations? |
---|---|---|---|---|---|
1 | How Many / Much / Heavy? Few? Seldom? Often? When? | Quantities, with Scale and a Zero Value.Differences and Ratios /Products are meaningful. | Quantitative/Ratio | Length,Height,Temperature in Kelvin,Activity,Dose Amount,Reaction Rate,Flow Rate,Concentration,Pulse,Survival Rate | Correlation |
4 | What, Who, Where, Whom, Which | Name, Place, Animal, Thing | Qualitative/Nominal | Name | Count no. of cases,Mode |
Inspiration
Alice said, “I say what I mean and I mean what I say!” Are the rest of us so sure? What do we mean when we use any of the phrases above? How definite are we? There is a range of “sureness” and “unsureness”…and this is where we can use box plots like Figure 1 to show that range of opinion.
Maybe it is time for a box plot on uh, shades1 of meaning for Jane Austen Gen-Z phrases! Bah.
How do these Chart(s) Work?
Box Plots are an extremely useful data visualization that gives us an idea of the distribution of a Quant variable, for each level of another Qual variable.
The internal process of this plot is as follows:
(Hat tip to student Tanya Michelle Justin for a good question on outlier calculation)
- Make groups of the Quant variable for each level of the Qual
- In each group, rank the Quant variable values in increasing order
- Calculate:
- The values for
median = Q2
, Q1, and Q3 based on rank!! -
Values for
min
,max
, and thenIQR = Q1 - Q3
- Calculate
outlier
limits:- \([Q1 - 1.5*IQR, Q2 + 1.5*IQR]\)
-
Whiskers
: All values within \([Q1 - 1.5*IQR, Q2 + 1.5*IQR]\) -
Outliers
: All values outside of \([Q1 - 1.5*IQR, Q2 + 1.5*IQR]\)
- The values for
- Plot these as a vertical or horizontal box structure, as shown.
As a result of this, while the box-part of the boxplot always shows 2 full quartiles, the whiskers may not stretch through their quartiles, since some values may be outliers on either side.
The Quant variable is ordered based on the values from min to max. So you could imagine that each value has a rank or sequence number. The min
value has \(rank = 1\) and the max
value has \(rank = length(var)\).
Note how the histogram that dwells upon the mean
and standard deviation
, whereas the boxplot focuses on the median
and quartiles
. The former uses the values of the Quant variable, whereas the latter uses their sequence number or ranks.
Box plots are often used for example in HR operations to understand Salary distributions across grades of employees. Marks of students in competitive exams are also declared using Quartiles.
Box plots can show skew in distributions. In such cases the “bottom” and the “lid” of the box may not be the same size!
In the Figure 3 (a), we see the difference between boxplots that show symmetric and skewed distributions. The “lid” and the “bottom” of the box are not of similar width in distributions with significant skewness.
Compare these with the corresponding Figure 3 (b).
Case Study-1: gss_wages
dataset
We will first look at Wage data from the General Social Survey (1974-2018) conducted in the USA, which is used to illustrate wage discrepancies by gender (while also considering respondent occupation, age, and education). This is available on Vincent Arel-Bundock’s superb repository of datasets. Let us read into R directly from the website.
wages <- read_csv("https://vincentarelbundock.github.io/Rdatasets/csv/stevedata/gss_wages.csv")
The data has automatically been read into the webr
session, so you can continue on to the next code chunk!
Examine the Data
As per our Workflow, we will look at the data using all the three methods we have seen.
glimpse(wages)
Rows: 61,697
Columns: 12
$ rownames <dbl> 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, …
$ year <dbl> 1974, 1974, 1974, 1974, 1974, 1974, 1974, 1974, 1974, 1974,…
$ realrinc <dbl> 4935, 43178, NA, NA, 18505, 22206, 55515, NA, NA, 4935, NA,…
$ age <dbl> 21, 41, 83, 69, 58, 30, 48, 67, 51, 54, 89, 71, 27, 30, 22,…
$ occ10 <dbl> 5620, 2040, NA, NA, 5820, 910, 230, 6355, 4720, 3940, 4810,…
$ occrecode <chr> "Office and Administrative Support", "Professional", NA, NA…
$ prestg10 <dbl> 25, 66, NA, NA, 37, 45, 59, 49, 28, 38, 47, 45, 50, 29, 33,…
$ childs <dbl> 0, 3, 2, 2, 0, 0, 2, 1, 2, 2, 3, 1, 4, 3, 0, 1, 2, 3, 4, 8,…
$ wrkstat <chr> "School", "Full-Time", "Housekeeper", "Housekeeper", "Full-…
$ gender <chr> "Male", "Male", "Female", "Female", "Female", "Male", "Male…
$ educcat <chr> "High School", "Bachelor", "Less Than High School", "Less T…
$ maritalcat <chr> "Married", "Married", "Widowed", "Widowed", "Never Married"…
skim(wages)
Name | wages |
Number of rows | 61697 |
Number of columns | 12 |
_______________________ | |
Column type frequency: | |
character | 5 |
numeric | 7 |
________________________ | |
Group variables | None |
Variable type: character
skim_variable | n_missing | complete_rate | min | max | empty | n_unique | whitespace |
---|---|---|---|---|---|---|---|
occrecode | 3561 | 0.94 | 5 | 37 | 0 | 11 | 0 |
wrkstat | 21 | 1.00 | 5 | 23 | 0 | 8 | 0 |
gender | 0 | 1.00 | 4 | 6 | 0 | 2 | 0 |
educcat | 135 | 1.00 | 8 | 21 | 0 | 5 | 0 |
maritalcat | 27 | 1.00 | 7 | 13 | 0 | 5 | 0 |
Variable type: numeric
skim_variable | n_missing | complete_rate | mean | sd | p0 | p25 | p50 | p75 | p100 | hist |
---|---|---|---|---|---|---|---|---|---|---|
rownames | 0 | 1.00 | 30849.00 | 17810.53 | 1 | 15425 | 30849 | 46273 | 61697.0 | ▇▇▇▇▇ |
year | 0 | 1.00 | 1996.07 | 12.79 | 1974 | 1985 | 1996 | 2006 | 2018.0 | ▆▇▇▇▇ |
realrinc | 23810 | 0.61 | 22326.36 | 28581.79 | 227 | 8156 | 16563 | 27171 | 480144.5 | ▇▁▁▁▁ |
age | 219 | 1.00 | 46.18 | 17.56 | 18 | 32 | 44 | 59 | 89.0 | ▇▇▆▅▂ |
occ10 | 3561 | 0.94 | 4695.77 | 2627.72 | 10 | 2710 | 4720 | 6230 | 9997.0 | ▃▅▇▂▃ |
prestg10 | 4186 | 0.93 | 43.06 | 12.99 | 16 | 33 | 42 | 50 | 80.0 | ▃▇▇▃▁ |
childs | 189 | 1.00 | 1.92 | 1.76 | 0 | 0 | 2 | 3 | 8.0 | ▇▇▂▁▁ |
inspect(wages)
categorical variables:
name class levels n missing
1 occrecode character 11 58136 3561
2 wrkstat character 8 61676 21
3 gender character 2 61697 0
4 educcat character 5 61562 135
5 maritalcat character 5 61670 27
distribution
1 Professional (19%), Service (16.9%) ...
2 Full-Time (49.4%), Housekeeper (15.1%) ...
3 Female (56.1%), Male (43.9%)
4 High School (51.5%) ...
5 Married (51.7%), Never Married (21.8%) ...
quantitative variables:
name class min Q1 median Q3 max mean sd
1 rownames numeric 1 15425 30849 46273 61697.0 30849.000000 17810.534116
2 year numeric 1974 1985 1996 2006 2018.0 1996.073715 12.794470
3 realrinc numeric 227 8156 16563 27171 480144.5 22326.359234 28581.794499
4 age numeric 18 32 44 59 89.0 46.176177 17.561065
5 occ10 numeric 10 2710 4720 6230 9997.0 4695.774081 2627.724076
6 prestg10 numeric 16 33 42 50 80.0 43.060701 12.987526
7 childs numeric 0 0 2 3 8.0 1.923457 1.763569
n missing
1 61697 0
2 61697 0
3 37887 23810
4 61478 219
5 58136 3561
6 57511 4186
7 61508 189
Data Dictionary
From the dataset documentation page, we note that this is a large dataset (61K rows), with 11 variables:
-
year
(dbl): the survey year -
realrinc
(dbl): the respondent’s base income (in constant 1986 USD -
age
(dbl): the respondent’s age in years -
occ10
(dbl): respondent’s occupation code (2010) -
prestg10
(dbl): respondent’s occupational prestige score (2010) -
childs
(dbl): number of children (0-8)
-
occrecode
(chr): recode of the occupation code into one of 11 main categories
-
wrkstat
(chr): the work status of the respondent (full-time, part-time, temporarily not working, unemployed (laid off), retired, school, housekeeper, other). 8 levels. -
gender
(chr): respondent’s gender (male or female). 2 levels.
-
educcat
(chr): respondent’s degree level (Less Than High School, High School, Junior College, Bachelor, or Graduate). 5 levels.
-
maritalcat
(chr): respondent’s marital status (Married, Widowed, Divorced, Separated, Never Married). 5 levels.
wages
dataset
- Fair amount of missing data; however with 61K rows, we can for the present simply neglect the missing data.
- Good mix of Qual and Quant variables
Hypothesis and Research Questions
- The target variable for an experiment that resulted in this data might be the
realinc
variable, the resultant income of the individual. Which is numerical variable.
- What is the basic distribution of
realrinc
? - Is
realrinc
affected bygender
? - By
educcat
? Bymaritalcat
? - Is
realrinc
affected bychild
? - Do combinations of these factors have an effect on the target variable?
These should do for now! But we should make more questions when have seen some plots!
Data Munging
Since there are so many missing data in the target variable realinc
and there is still enough data leftover, let us remove the rows containing missing data in that variable.
NOTE: This is not advised at all as a general procedure!! Data is valuable and there are better ways to manage this problem!
Plotting Box Plots
Question-1: What is the basic distribution of realrinc
?
realrinc
?
# Set graph theme
theme_set(new = theme_custom())
#
wages_clean %>%
gf_boxplot(realrinc ~ "Income") %>% # Dummy X-axis "variable"
gf_labs(
title = "Plot 1A: Income has a skewed distribution",
subtitle = "Many outliers on the high side"
)
Business Insights-1
- Income is a very skewed distribution, as might be expected.
- Presence of many higher-side outliers is noted.
Question-2: Is realrinc
affected by gender
?
realrinc
affected by gender
?
# Set graph theme
theme_set(new = theme_custom())
#
wages_clean %>%
gf_boxplot(gender ~ realrinc) %>%
gf_labs(title = "Plot 2A: Income by Gender")
# Set graph theme
theme_set(new = theme_custom())
wages_clean %>%
gf_boxplot(gender ~ realrinc, fill = ~gender) %>%
gf_refine(scale_x_log10()) %>%
gf_labs(title = "Plot 2C: Income filled by Gender, log scale")
# Set graph theme
theme_set(new = theme_custom())
#
wages_clean %>%
ggplot() +
geom_boxplot(aes(y = gender, x = realrinc)) +
labs(title = "Plot 2A: Income by Gender")
##
wages_clean %>%
ggplot() +
geom_boxplot(aes(y = gender, x = log10(realrinc))) +
labs(title = "Plot 2B: Log(Income) by Gender")
##
wages_clean %>%
ggplot() +
geom_boxplot(aes(y = gender, x = realrinc, fill = gender)) +
gf_refine(scale_x_log10()) +
labs(title = "Plot 2C: Income filled by Gender, log scale")
Business Insights-2
- Even when split by
gender
,realincome
presents a skewed set of distributions. - The IQR for
male
s is smaller than the IQR forfemale
s. There is less variation in the middle ranges ofrealrinc
for men. - log10 transformation helps to view and understand the regions of low
realrinc
. - There are outliers on both sides, indicating that there may be many people who make very small amounts of money and large amounts of money in both
gender
s.
Question-3: Is realrinc
affected by educcat
?
realrinc
affected by educcat
?
# Set graph theme
theme_set(new = theme_custom())
#
wages_clean %>%
gf_boxplot(educcat ~ realrinc) %>%
gf_labs(title = "Plot 3A: Income by Education Category")
# Set graph theme
theme_set(new = theme_custom())
wages_clean %>%
gf_boxplot(reorder(educcat, realrinc, FUN = median) ~ realrinc,
fill = ~educcat,
alpha = 0.5
) %>%
gf_refine(scale_x_log10()) %>%
gf_labs(
title = "Plot 3D: Income by Education Category, sorted",
subtitle = "Log Income"
) %>%
gf_labs(
x = "Income",
y = "Education Category"
)
# Set graph theme
theme_set(new = theme_custom())
#
wages_clean %>%
ggplot() +
geom_boxplot(aes(realrinc, educcat)) + # (x,y) format
labs(title = "Plot 3A: Income by Education Category")
##
wages_clean %>%
ggplot() +
geom_boxplot(aes(log10(realrinc), educcat)) +
labs(title = "Plot 3B: Log(Income) by Education Category")
##
wages_clean %>%
ggplot() +
geom_boxplot(
aes(log(realrinc),
reorder(educcat, realrinc, FUN = median),
fill = educcat
),
alpha = 0.3
) +
labs(
title = "Plot 3C: Log(Income) by Education Category, sorted",
x = "Log Income", y = "Education Category"
)
##
wages_clean %>%
ggplot() +
geom_boxplot(
aes(realrinc,
reorder(educcat, realrinc, FUN = median),
fill = educcat
),
alpha = 0.3
) +
scale_x_log10() +
labs(
title = "Plot 3D: Income by Education Category, sorted",
subtitle = "Log Income Scale",
x = "Income", y = "Education Category"
)
Business Insights-3
-
realrinc
rises witheduccat
, which is to be expected. - However, there are people with very low and very high income in all categories of
educcat
- Hence
educcat
alone may not be a good predictor forrealrinc
.
We can do similar work with the other Qual variables. Let us now see how we can use more than one Qual variable and answer the last hypothesis, Question 4.
Question-4: Is the target variable realrinc
affected by combinations of Qual factors gender
, educcat
, maritalcat
and childs
?
This is a rather complex question and could take us deep into Modelling. Ideally we ought to:
- take each Qual variable, explain its effect on the target variable
-
remove that effect and model the remainder ( i.e. residual) with the next Qual variable
- Proceed in this way until we have a good model.
if we are going to do this manually.
There are more modern Modelling Workflows, that can do things much faster and without such manual tweaking.
So will simply plot box plots showing effects on the target variable of combinations of Qual variables taken two at a time. (We will of course use facetted box plots!)
We will also drop NA values all around this time, to avoid seeing boxplots for undocumented categories.
realrinc
affected by combinations of factors?
# Set graph theme
theme_set(new = theme_custom())
#
wages %>%
drop_na() %>%
gf_boxplot(reorder(educcat, realrinc) ~ log10(realrinc),
fill = ~educcat,
alpha = 0.5
) %>%
gf_facet_wrap(vars(childs)) %>%
gf_refine(scale_fill_brewer(type = "qual", palette = "Dark2")) %>%
gf_labs(
title = "Plot 4A: Log Income by Education Category and Family Size",
x = "Log income",
y = "No. of Children"
)
# Set graph theme
theme_set(new = theme_custom())
#
wages %>%
drop_na() %>%
mutate(childs = as_factor(childs)) %>%
gf_boxplot(childs ~ log10(realrinc),
group = ~childs,
fill = ~childs,
alpha = 0.5
) %>%
gf_facet_wrap(~gender) %>%
gf_refine(scale_fill_brewer(type = "qual", palette = "Set3")) %>%
gf_labs(
title = "Plot 4B: Log Income by Gender and Family Size",
x = "Log income",
y = "No. of Children"
)
# Set graph theme
theme_set(new = theme_custom())
#
wages %>%
drop_na() %>%
ggplot() +
geom_boxplot(
aes(log10(realrinc), reorder(educcat, realrinc),
fill = educcat
), # aes() closes here
alpha = 0.5
) +
facet_wrap(vars(childs)) +
scale_fill_brewer(type = "qual", palette = "Dark2") +
labs(title = "Plot 4A: Log Income by Education Category and Family Size", x = "Log income", y = "No. of Children")
##
wages %>%
drop_na() %>%
mutate(childs = as_factor(childs)) %>%
ggplot() +
geom_boxplot(
aes(log10(realrinc), childs,
group = childs,
fill = childs
), # aes() closes here
alpha = 0.5
) +
facet_wrap(vars(gender)) +
scale_fill_brewer(type = "qual", palette = "Set3") +
labs(
title = "Plot 4B: Log Income by Gender and Family Size",
x = "Log income",
y = "No. of Children"
)
Business Insights-4
- From Figure 6, we see that
realrinc
increases witheduccat
, across (almost) all family sizeschilds
. - However, this trend breaks a little when family sizes
childs
is large, say >= 7. Be aware that the data observations for such large families may be sparse and this inference may not be necessarily valid. - From Figure 7, we see that the effect of
childs
onrealrinc
is different for eachgender
! For females, the income steadily drops with the number of children, whereas for males it actually increases up to a certain family size before decreasing again.
Are the Differences Significant?
In data analysis, we always want to know2, as in life, how important things are, whether they matter. To do this, we make up hunches, or more precisely, Hypotheses. We make two in fact:
\(H_0\): Nothing is happening;
\(H_a\): (“a” for Alternate): Something is happening and it is important enough to pay attention to.
We then pretend that \(H_0\) is true and ask that our data prove us wrong; if it does, we reject \(H_0\) in favour of \(H_a\).
This is a very important idea of Hypothesis Testing which helps you justify your hunch. We will study this when we do Stats Tests for differences between two means(t-tests), and those between more than two means(ANOVA).
Wait, But Why?
- Box plots are a powerful statistical graphic that give us a combined view of data ranges, quartiles, medians, and outliers.
- Box plots can compare
groups
within our Quant variable, based onlevels
of a Qual variable. This is a very common and important task in research! - In your design research, you would have numerical Quant data that is accompanied by categorical Qual data pertaining to groups within your target audience.
- Analyzing for differences in the Quant across levels of the Qual (e.g household expenditure across groups of people) is a vital step in justifying time, effort, and money for further actions in your project. Don’t faff this.
- Box plots are ideal for visualizing statistical tests for difference in mean values across groups (t-test and ANOVA). (Even though they plot medians)
Conclusion
- Box Plots “dwell upon” medians and Quartiles
- Box Plots can show distributions of a Quant variable over levels of a Qual variable
- This allows a comparison of box plots side by side to visibly detect differences in medians and IQRs across such levels.
Your Turn
Here are a couple of datasets that you might want to analyze with box plots:
Rows: 80332 Columns: 11
── Column specification ────────────────────────────────────────────────────────
Delimiter: ","
chr (8): date_time, city_area, state, country, ufo_shape, described_encounte...
dbl (3): encounter_length, latitude, longitude
ℹ Use `spec()` to retrieve the full column specification for this data.
ℹ Specify the column types or set `show_col_types = FALSE` to quiet this message.