Chapter 4 Data wrangling 1
In this lecture, we will take a look at how to wrangle data using the dplyr package. Again, getting our data into shape is something we’ll need to do throughout the course, so it’s worth spending some time getting a good sense for how this works. The nice thing about R is that (thanks to the tidyverse
), both visualization and data wrangling are particularly powerful.
4.1 Learning goals
- Review R basics (incl. variable modes, data types, operators, control flow, and functions).
- Learn how the pipe operator
%>%
works. - See different ways for getting a sense of one’s data.
- Master key data manipulation verbs from the
dplyr
package (incl.filter()
,arrange()
,rename()
,relocate()
,select()
,mutate()
) as well as the helper functionsacross()
andwhere()
.
4.3 Some R basics
To test your knowledge of the R basics, I recommend taking the free interactive tutorial on datacamp: Introduction to R. Here, I will just give a very quick overview of some of the basics.
4.3.1 Modes
Variables in R can have different modes. Table 4.1 shows the most common ones.
name | example |
---|---|
numeric | 1 , 3 , 48 |
character | 'Steve' , 'a' , '78' |
logical | TRUE , FALSE |
factor | 'small' , 'medium' , 'large' |
not available | NA |
For characters you can either use "
or '
. R has a number of functions to convert a variable from one mode to another. NA
is used for missing values.
chr "1"
num 1
Factor w/ 1 level "1": 1
chr "1"
[1] TRUE
The str()
function displays the structure of an R object. Here, it shows us what mode the variable is.
4.3.2 Data types
R has a number of different data types. Table 4.2 shows the ones you’re most likely to come across (taken from this source):
name | description |
---|---|
vector | list of values with of the same variable mode |
matrix | 2D data structure |
array | same as matrix for higher dimensional data |
data frame | similar to matrix but with column names |
list | flexible type that can contain different other variable types |
4.3.2.1 Vectors
We build vectors using the concatenate function c()
, and we use []
to access one or more elements of a vector.
[1] 4
[1] 1 4
[1] 1 5
In R (unlike in Python for example), 1 refers to the first element of a vector (or list).
4.3.2.2 Matrix
We build a matrix using the matrix()
function, and we use []
to access its elements.
[,1] [,2]
[1,] 1 4
[2,] 2 5
[3,] 3 6
[1] 4
[1] 1 4
[1] 1 2 3
[,1] [,2]
[1,] 2 5
[2,] 3 6
Note how we use an empty placeholder to indicate that we want to select all the values in a row or column, and -
to indicate that we want to remove something.
4.3.2.4 Data frame
df = tibble(participant_id = c(1, 2, 3),
participant_name = c("Leia", "Luke", "Darth")) # make the data frame
df # the complete data frame
# A tibble: 3 × 2
participant_id participant_name
<dbl> <chr>
1 1 Leia
2 2 Luke
3 3 Darth
# A tibble: 1 × 1
participant_name
<chr>
1 Leia
[1] 1 2 3
[1] 1 2 3
[1] "Luke"
[1] "Luke"
We’ll use data frames a lot. Data frames are like a matrix with column names. Data frames are also more general than matrices in that different columns can have different modes. For example, one column might be a character, another one numeric, and another one a factor.
Here we used the tibble()
function to create the data frame. A tibble
is almost the same as a data frame but it has better defaults for formatting output in the console (more information on tibbles is here).
4.3.2.5 Lists
l.mixed = list(number = 1,
character = "2",
factor = factor(3),
matrix = matrix(1:4, ncol = 2),
df = tibble(x = c(1, 2), y = c(3, 4)))
l.mixed
$number
[1] 1
$character
[1] "2"
$factor
[1] 3
Levels: 3
$matrix
[,1] [,2]
[1,] 1 3
[2,] 2 4
$df
# A tibble: 2 × 2
x y
<dbl> <dbl>
1 1 3
2 2 4
[1] "2"
[1] "2"
[1] "2"
Lists are a very flexible data format. You can put almost anything in a list.
4.3.3 Operators
Table 4.3 shows the comparison operators that result in logical outputs.
symbol | name |
---|---|
== |
equal to |
!= |
not equal to |
> , < |
greater/less than |
>= , <= |
greater/less than or equal |
& , |, ! |
logical operators: and, or, not |
%in% |
checks whether an element is in an object |
The %in%
operator is very useful, and we can use it like so:
[1] TRUE
[1] TRUE FALSE
It’s particularly useful for filtering data as we will see below.
4.3.4 Control flow
4.3.4.1 if-then
number = 3
if (number == 1) {
print("The number is 1.")
} else if (number == 2) {
print("The number is 2.")
} else {
print("The number is neither 1 nor 2.")
}
[1] "The number is neither 1 nor 2."
As a shorthand version, we can also use the ifelse()
function like so:
[1] "false"
4.3.5 Functions
fun.add_two_numbers = function(a, b){
x = a + b
return(str_c("The result is ", x))
}
fun.add_two_numbers(1, 2)
[1] "The result is 3"
I’ve used the str_c()
function here to concatenate the string with the number. (R converts the number x
into a string for us.) Note, R functions can only return a single object. However, this object can be a list (which can contain anything).
4.3.5.1 Some often used functions
name | description |
---|---|
length() |
length of an object |
dim() |
dimensions of an object (e.g. number of rows and columns) |
rm() |
remove an object |
seq() |
generate a sequence of numbers |
rep() |
repeat something n times |
max() |
maximum |
min() |
minimum |
which.max() |
index of the maximum |
which.min() |
index of the maximum |
mean() |
mean |
median() |
median |
sum() |
sum |
var() |
variance |
sd() |
standard deviation |
4.3.6 The pipe operator %>%
The pipe operator %>%
is a special operator introduced in the magrittr
package. It is used heavily in the tidyverse. The basic idea is simple: this operator allows us to “pipe” several functions into one long chain that matches the order in which we want to do stuff.
Let’s consider the following example of making and eating a cake (thanks to https://twitter.com/dmi3k/status/1191824875842879489?s=09). This would be the traditional way of writing some code:
eat(
slice(
bake(
put(
pour(
mix(ingredients),
into = baking_form),
into = oven),
time = 30),
pieces = 6),
1)
To see what’s going on here, we need to read the code inside out. That is, we have to start in the innermost bracket, and then work our way outward. However, there is a natural causal ordering to these steps and wouldn’t it be nice if we could just write code in that order? Thanks to the pipe operator %>%
we can! Here is the same example using the pipe:
ingredients %>%
mix %>%
pour(into = baking_form) %>%
put(into = oven) %>%
bake(time = 30) %>%
slice(pieces = 6) %>%
eat(1)
This code is much easier to read and write, since it represents the order in which we want to do things!
Abstractly, the pipe operator does the following:
f(x)
can be rewritten asx %>% f()
For example, in standard R, we would write:
[1] 6
With the pipe, we can rewrite this as:
[1] 6
This doesn’t seem super useful yet, but just hold on a little longer.
f(x, y)
can be rewritten asx %>% f(y)
So, we could rewrite the following standard R code …
[1] 3.141593
… by using the pipe:
[1] 3.141593
Here is another example:
[1] 7
[1] 7
The pipe operator inserts the result of the previous computation as a first element into the next computation. So, a %>% sum(b)
is equivalent to sum(a, b)
. We can also specify to insert the result at a different position via the .
operator. For example:
[1] 1 2 3 4 5 6 7 8 9 10
Here, I used the .
operator to specify that I woud like to insert the result of b
where I’ve put the .
in the seq()
function.
f(x, y)
can be rewritten asy %>% f(x, .)
Still not to thrilled about the pipe? We can keep going though (and I’m sure you’ll be convinced eventually.)
h(g(f(x)))
can be rewritten asx %>% f() %>% g() %>% h()
For example, consider that we want to calculate the root mean squared error (RMSE) between prediction and data.
Here is how the RMSE is defined:
\[ \text{RMSE} = \sqrt\frac{\sum_{i=1}^n(\hat{y}_i-y_i)^2}{n} \] where \(\hat{y}_i\) denotes the prediction, and \(y_i\) the actually observed value.
In base R, we would do the following.
data = c(1, 3, 4, 2, 5)
prediction = c(1, 2, 2, 1, 4)
# calculate root mean squared error
rmse = sqrt(mean((prediction-data)^2))
print(rmse)
[1] 1.183216
Using the pipe operator makes the operation more intuitive:
data = c(1, 3, 4, 2, 5)
prediction = c(1, 2, 2, 1, 4)
# calculate root mean squared error the pipe way
rmse = (prediction-data)^2 %>%
mean() %>%
sqrt() %>%
print()
[1] 1.183216
First, we calculate the squared error, then we take the mean, then the square root, and then print the result.
The pipe operator %>%
is similar to the +
used in ggplot2
. It allows us to take step-by-step actions in a way that fits the causal ordering of how we want to do things.
Tip: The keyboard shortcut for the pipe operator is:
cmd/ctrl + shift + m
Definitely learn this one – we’ll use the pipe a lot!!
Tip: Code is generally easier to read when the pipe
%>%
is at the end of a line (just like the+
inggplot2
).
A key advantage of using the pipe is that you don’t have to save intermediate computations as new variables and this helps to keep your environment nice and clean!
4.4 A quick note on naming things
Personally, I like to name things in a (pretty) consistent way so that I have no trouble finding stuff even when I open up a project that I haven’t worked on for a while. I try to use the following naming conventions:
name | use |
---|---|
df.thing | for data frames |
l.thing | for lists |
fun.thing | for functions |
tmp.thing | for temporary variables |
4.5 Looking at data
The package dplyr
which we loaded as part of the tidyverse, includes a data set with information about starwars characters. Let’s store this as df.starwars
.
Note: Unlike in other languages (such as Python or Matlab), a
.
in a variable name has no special meaning and can just be used as part of the name. I’ve useddf
here to indicate for myself that this variable is a data frame. Before visualizing the data, it’s often useful to take a quick direct look at the data.
There are several ways of taking a look at data in R. Personally, I like to look at the data within RStudio’s data viewer. To do so, you can:
- click on the
df.starwars
variable in the “Environment” tab
- type
View(df.starwars)
in the console - move your mouse over (or select) the variable in the editor (or console) and hit
F2
I like the F2
route the best as it’s fast and flexible.
Sometimes it’s also helpful to look at data in the console instead of the data viewer. Particularly when the data is very large, the data viewer can be sluggish.
Here are some useful functions:
4.5.1 head()
Without any extra arguments specified, head()
shows the top six rows of the data.
# A tibble: 6 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Luke Sky… 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth Va… 202 136 none white yellow 41.9 male mascu…
5 Leia Org… 150 49 brown light brown 19 fema… femin…
6 Owen Lars 178 120 brown, gr… light blue 52 male mascu…
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
4.5.2 glimpse()
glimpse()
is helpful when the data frame has many columns. The data is shown in a transposed way with columns as rows.
Rows: 87
Columns: 14
$ name <chr> "Luke Skywalker", "C-3PO", "R2-D2", "Darth Vader", "Leia Or…
$ height <int> 172, 167, 96, 202, 150, 178, 165, 97, 183, 182, 188, 180, 2…
$ mass <dbl> 77.0, 75.0, 32.0, 136.0, 49.0, 120.0, 75.0, 32.0, 84.0, 77.…
$ hair_color <chr> "blond", NA, NA, "none", "brown", "brown, grey", "brown", N…
$ skin_color <chr> "fair", "gold", "white, blue", "white", "light", "light", "…
$ eye_color <chr> "blue", "yellow", "red", "yellow", "brown", "blue", "blue",…
$ birth_year <dbl> 19.0, 112.0, 33.0, 41.9, 19.0, 52.0, 47.0, NA, 24.0, 57.0, …
$ sex <chr> "male", "none", "none", "male", "female", "male", "female",…
$ gender <chr> "masculine", "masculine", "masculine", "masculine", "femini…
$ homeworld <chr> "Tatooine", "Tatooine", "Naboo", "Tatooine", "Alderaan", "T…
$ species <chr> "Human", "Droid", "Droid", "Human", "Human", "Human", "Huma…
$ films <list> <"A New Hope", "The Empire Strikes Back", "Return of the J…
$ vehicles <list> <"Snowspeeder", "Imperial Speeder Bike">, <>, <>, <>, "Imp…
$ starships <list> <"X-wing", "Imperial shuttle">, <>, <>, "TIE Advanced x1",…
4.5.3 distinct()
distinct()
shows all the distinct values for a character or factor column.
# A tibble: 38 × 1
species
<chr>
1 Human
2 Droid
3 Wookiee
4 Rodian
5 Hutt
6 <NA>
7 Yoda's species
8 Trandoshan
9 Mon Calamari
10 Ewok
# ℹ 28 more rows
4.5.4 count()
count()
shows a count of all the different distinct values in a column.
# A tibble: 15 × 2
eye_color n
<chr> <int>
1 black 10
2 blue 19
3 blue-gray 1
4 brown 21
5 dark 1
6 gold 1
7 green, yellow 1
8 hazel 3
9 orange 8
10 pink 1
11 red 5
12 red, blue 1
13 unknown 3
14 white 1
15 yellow 11
It’s possible to do grouped counts by combining several variables.
# A tibble: 10 × 3
eye_color gender n
<chr> <chr> <int>
1 black feminine 2
2 black masculine 8
3 blue feminine 6
4 blue masculine 12
5 blue <NA> 1
6 blue-gray masculine 1
7 brown feminine 4
8 brown masculine 15
9 brown <NA> 2
10 dark masculine 1
4.5.5 datatable()
For RMardkown files specifically, we can use the datatable()
function from the DT
package to get an interactive table widget.
4.5.6 Other tools for taking a quick look at data
4.5.6.1 vis_dat()
The vis_dat()
function from the visdat
package, gives a visual summary that makes it easy to see the variable types and whether there are missing values in the data.
When R loads packages, functions loaded in earlier packages are
overwritten by functions of the same name from later packages. This
means that the order in which packages are loaded matters. To make sure
that a function from the correct package is used, you can use the
package_name::function_name()
construction. This way, the
function_name()
from the package_name
is used,
rather than the same function from a different package.
This is why, in general, I recommend to load the tidyverse package last (since it contains a large number of functions that we use a lot).
4.5.6.2 skim()
The skim()
function from the skimr
package provides a nice overview of the data, separated by variable types.
Name | df.starwars |
Number of rows | 87 |
Number of columns | 14 |
_______________________ | |
Column type frequency: | |
character | 8 |
list | 3 |
numeric | 3 |
________________________ | |
Group variables | None |
Variable type: character
skim_variable | n_missing | complete_rate | min | max | empty | n_unique | whitespace |
---|---|---|---|---|---|---|---|
name | 0 | 1.00 | 3 | 21 | 0 | 87 | 0 |
hair_color | 5 | 0.94 | 4 | 13 | 0 | 11 | 0 |
skin_color | 0 | 1.00 | 3 | 19 | 0 | 31 | 0 |
eye_color | 0 | 1.00 | 3 | 13 | 0 | 15 | 0 |
sex | 4 | 0.95 | 4 | 14 | 0 | 4 | 0 |
gender | 4 | 0.95 | 8 | 9 | 0 | 2 | 0 |
homeworld | 10 | 0.89 | 4 | 14 | 0 | 48 | 0 |
species | 4 | 0.95 | 3 | 14 | 0 | 37 | 0 |
Variable type: list
skim_variable | n_missing | complete_rate | n_unique | min_length | max_length |
---|---|---|---|---|---|
films | 0 | 1 | 24 | 1 | 7 |
vehicles | 0 | 1 | 11 | 0 | 2 |
starships | 0 | 1 | 16 | 0 | 5 |
Variable type: numeric
skim_variable | n_missing | complete_rate | mean | sd | p0 | p25 | p50 | p75 | p100 | hist |
---|---|---|---|---|---|---|---|---|---|---|
height | 6 | 0.93 | 174.60 | 34.77 | 66 | 167.0 | 180 | 191.0 | 264 | ▂▁▇▅▁ |
mass | 28 | 0.68 | 97.31 | 169.46 | 15 | 55.6 | 79 | 84.5 | 1358 | ▇▁▁▁▁ |
birth_year | 44 | 0.49 | 87.57 | 154.69 | 8 | 35.0 | 52 | 72.0 | 896 | ▇▁▁▁▁ |
4.5.6.3 dfSummary()
The summarytools
package is another great package for taking a look at the data. It renders a nice html output for the data frame including a lot of helpful information. You can find out more about this package here.
df.starwars %>%
select(where(~ !is.list(.))) %>% # this removes all list columns
summarytools::dfSummary() %>%
summarytools::view()
Note: The summarytools::view() function will not show up here in the html. It generates a summary of the data that is displayed in the Viewer in RStudio.
Once we’ve taken a look at the data, the next step would be to visualize relationships between variables of interest.
4.6 Wrangling data
We use the functions in the package dplyr
to manipulate our data.
4.6.1 filter()
filter()
lets us apply logical (and other) operators (see Table 4.3) to subset the data. Here, I’ve filtered out the masculine characters.
# A tibble: 66 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Luke Sk… 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth V… 202 136 none white yellow 41.9 male mascu…
5 Owen La… 178 120 brown, gr… light blue 52 male mascu…
6 R5-D4 97 32 <NA> white, red red NA none mascu…
7 Biggs D… 183 84 black light brown 24 male mascu…
8 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
9 Anakin … 188 84 blond fair blue 41.9 male mascu…
10 Wilhuff… 180 NA auburn, g… fair blue 64 male mascu…
# ℹ 56 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
We can combine multiple conditions in the same call. Here, I’ve filtered out masculine characters, whose height is greater than the median height (i.e. they are in the top 50 percentile), and whose mass was not NA
.
# A tibble: 26 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Darth V… 202 136 none white yellow 41.9 male mascu…
2 Biggs D… 183 84 black light brown 24 male mascu…
3 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
4 Anakin … 188 84 blond fair blue 41.9 male mascu…
5 Chewbac… 228 112 brown unknown blue 200 male mascu…
6 Boba Fe… 183 78.2 black fair brown 31.5 male mascu…
7 IG-88 200 140 none metal red 15 none mascu…
8 Bossk 190 113 none green red 53 male mascu…
9 Qui-Gon… 193 89 brown fair blue 92 male mascu…
10 Nute Gu… 191 90 none mottled g… red NA male mascu…
# ℹ 16 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
Many functions like mean()
, median()
, var()
, sd()
, sum()
have the argument na.rm
which is set to FALSE
by default. I set the argument to TRUE
here (or T
for short), which means that the NA
values are ignored, and the median()
is calculated based on the remaining values.
You can use ,
and &
interchangeably in filter()
. Make sure to use parentheses when combining several logical operators to indicate which logical operation should be performed first:
df.starwars %>%
filter((skin_color %in% c("dark", "pale") | sex == "hermaphroditic") & height > 170)
# A tibble: 10 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Jabba D… 175 1358 <NA> green-tan… orange 600 herm… mascu…
2 Lando C… 177 79 black dark brown 31 male mascu…
3 Quarsh … 183 NA black dark brown 62 male mascu…
4 Bib For… 180 NA none pale pink NA male mascu…
5 Mace Wi… 188 84 none dark brown 72 male mascu…
6 Ki-Adi-… 198 82 white pale yellow 92 male mascu…
7 Adi Gal… 184 50 none dark blue NA fema… femin…
8 Saesee … 188 NA none pale orange NA male mascu…
9 Gregar … 185 85 black dark brown NA <NA> <NA>
10 Sly Moo… 178 48 none pale white NA <NA> <NA>
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
The starwars characters that have either a "dark"
or a "pale"
skin tone, or whose sex is "hermaphroditic"
, and whose height is at least 170
cm. The %in%
operator is useful when there are multiple options. Instead of skin_color %in% c("dark", "pale")
, I could have also written skin_color == "dark" | skin_color == "pale"
but this gets cumbersome as the number of options increases.
4.6.2 arrange()
arrange()
allows us to sort the values in a data frame by one or more column entries.
# A tibble: 87 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Mon Mot… 150 NA auburn fair blue 48 fema… femin…
2 Wilhuff… 180 NA auburn, g… fair blue 64 male mascu…
3 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
4 Bail Pr… 191 NA black tan brown 67 male mascu…
5 Gregar … 185 85 black dark brown NA <NA> <NA>
6 Biggs D… 183 84 black light brown 24 male mascu…
7 Boba Fe… 183 78.2 black fair brown 31.5 male mascu…
8 Quarsh … 183 NA black dark brown 62 male mascu…
9 Jango F… 183 79 black tan brown 66 male mascu…
10 Lando C… 177 79 black dark brown 31 male mascu…
# ℹ 77 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
Here, I’ve sorted the data frame first by hair_color
, and then by height
. I’ve used the desc()
function to sort height
in descending order. Bail Prestor Organa is the tallest black character in starwars.
4.6.3 rename()
rename()
renames column names.
# A tibble: 87 × 14
person height mass_kg hair_color skin_color eye_color birth_year sex gender
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Luke … 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth… 202 136 none white yellow 41.9 male mascu…
5 Leia … 150 49 brown light brown 19 fema… femin…
6 Owen … 178 120 brown, gr… light blue 52 male mascu…
7 Beru … 165 75 brown light blue 47 fema… femin…
8 R5-D4 97 32 <NA> white, red red NA none mascu…
9 Biggs… 183 84 black light brown 24 male mascu…
10 Obi-W… 182 77 auburn, w… fair blue-gray 57 male mascu…
# ℹ 77 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
The new variable names goes on the LHS of the=
sign, and the old name on the RHS.
To rename all variables at the same time use rename_with()
:
# A tibble: 87 × 14
NAME HEIGHT MASS HAIR_COLOR SKIN_COLOR EYE_COLOR BIRTH_YEAR SEX GENDER
<chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Luke Sk… 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth V… 202 136 none white yellow 41.9 male mascu…
5 Leia Or… 150 49 brown light brown 19 fema… femin…
6 Owen La… 178 120 brown, gr… light blue 52 male mascu…
7 Beru Wh… 165 75 brown light blue 47 fema… femin…
8 R5-D4 97 32 <NA> white, red red NA none mascu…
9 Biggs D… 183 84 black light brown 24 male mascu…
10 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
# ℹ 77 more rows
# ℹ 5 more variables: HOMEWORLD <chr>, SPECIES <chr>, FILMS <list>,
# VEHICLES <list>, STARSHIPS <list>
Notice that I used the ~
here in the function call. I will explain what this does shortly.
4.6.4 relocate()
relocate()
moves columns. For example, the following piece of code moves the species
column to the front of the data frame:
# A tibble: 87 × 14
species name height mass hair_color skin_color eye_color birth_year sex
<chr> <chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr>
1 Human Luke S… 172 77 blond fair blue 19 male
2 Droid C-3PO 167 75 <NA> gold yellow 112 none
3 Droid R2-D2 96 32 <NA> white, bl… red 33 none
4 Human Darth … 202 136 none white yellow 41.9 male
5 Human Leia O… 150 49 brown light brown 19 fema…
6 Human Owen L… 178 120 brown, gr… light blue 52 male
7 Human Beru W… 165 75 brown light blue 47 fema…
8 Droid R5-D4 97 32 <NA> white, red red NA none
9 Human Biggs … 183 84 black light brown 24 male
10 Human Obi-Wa… 182 77 auburn, w… fair blue-gray 57 male
# ℹ 77 more rows
# ℹ 5 more variables: gender <chr>, homeworld <chr>, films <list>,
# vehicles <list>, starships <list>
We could also move the species
column after the name column like so:
# A tibble: 87 × 14
name species height mass hair_color skin_color eye_color birth_year sex
<chr> <chr> <int> <dbl> <chr> <chr> <chr> <dbl> <chr>
1 Luke S… Human 172 77 blond fair blue 19 male
2 C-3PO Droid 167 75 <NA> gold yellow 112 none
3 R2-D2 Droid 96 32 <NA> white, bl… red 33 none
4 Darth … Human 202 136 none white yellow 41.9 male
5 Leia O… Human 150 49 brown light brown 19 fema…
6 Owen L… Human 178 120 brown, gr… light blue 52 male
7 Beru W… Human 165 75 brown light blue 47 fema…
8 R5-D4 Droid 97 32 <NA> white, red red NA none
9 Biggs … Human 183 84 black light brown 24 male
10 Obi-Wa… Human 182 77 auburn, w… fair blue-gray 57 male
# ℹ 77 more rows
# ℹ 5 more variables: gender <chr>, homeworld <chr>, films <list>,
# vehicles <list>, starships <list>
4.6.5 select()
select()
allows us to select a subset of the columns in the data frame.
# A tibble: 87 × 3
name height mass
<chr> <int> <dbl>
1 Luke Skywalker 172 77
2 C-3PO 167 75
3 R2-D2 96 32
4 Darth Vader 202 136
5 Leia Organa 150 49
6 Owen Lars 178 120
7 Beru Whitesun Lars 165 75
8 R5-D4 97 32
9 Biggs Darklighter 183 84
10 Obi-Wan Kenobi 182 77
# ℹ 77 more rows
We can select multiple columns using the (from:to)
syntax:
# A tibble: 87 × 7
name height mass hair_color skin_color eye_color birth_year
<chr> <int> <dbl> <chr> <chr> <chr> <dbl>
1 Luke Skywalker 172 77 blond fair blue 19
2 C-3PO 167 75 <NA> gold yellow 112
3 R2-D2 96 32 <NA> white, bl… red 33
4 Darth Vader 202 136 none white yellow 41.9
5 Leia Organa 150 49 brown light brown 19
6 Owen Lars 178 120 brown, grey light blue 52
7 Beru Whitesun Lars 165 75 brown light blue 47
8 R5-D4 97 32 <NA> white, red red NA
9 Biggs Darklighter 183 84 black light brown 24
10 Obi-Wan Kenobi 182 77 auburn, white fair blue-gray 57
# ℹ 77 more rows
Or use a variable for column selection:
columns = c("name", "height", "species")
df.starwars %>%
select(one_of(columns)) # useful when using a variable for column selection
# A tibble: 87 × 3
name height species
<chr> <int> <chr>
1 Luke Skywalker 172 Human
2 C-3PO 167 Droid
3 R2-D2 96 Droid
4 Darth Vader 202 Human
5 Leia Organa 150 Human
6 Owen Lars 178 Human
7 Beru Whitesun Lars 165 Human
8 R5-D4 97 Droid
9 Biggs Darklighter 183 Human
10 Obi-Wan Kenobi 182 Human
# ℹ 77 more rows
We can also deselect (multiple) columns:
# A tibble: 87 × 6
height mass hair_color skin_color eye_color starships
<int> <dbl> <chr> <chr> <chr> <list>
1 172 77 blond fair blue <chr [2]>
2 167 75 <NA> gold yellow <chr [0]>
3 96 32 <NA> white, blue red <chr [0]>
4 202 136 none white yellow <chr [1]>
5 150 49 brown light brown <chr [0]>
6 178 120 brown, grey light blue <chr [0]>
7 165 75 brown light blue <chr [0]>
8 97 32 <NA> white, red red <chr [0]>
9 183 84 black light brown <chr [1]>
10 182 77 auburn, white fair blue-gray <chr [5]>
# ℹ 77 more rows
And select columns by partially matching the column name:
# A tibble: 87 × 4
hair_color skin_color eye_color birth_year
<chr> <chr> <chr> <dbl>
1 blond fair blue 19
2 <NA> gold yellow 112
3 <NA> white, blue red 33
4 none white yellow 41.9
5 brown light brown 19
6 brown, grey light blue 52
7 brown light blue 47
8 <NA> white, red red NA
9 black light brown 24
10 auburn, white fair blue-gray 57
# ℹ 77 more rows
# A tibble: 87 × 3
height hair_color homeworld
<int> <chr> <chr>
1 172 blond Tatooine
2 167 <NA> Tatooine
3 96 <NA> Naboo
4 202 none Tatooine
5 150 brown Alderaan
6 178 brown, grey Tatooine
7 165 brown Tatooine
8 97 <NA> Tatooine
9 183 black Tatooine
10 182 auburn, white Stewjon
# ℹ 77 more rows
We can rename some of the columns using select()
like so:
# A tibble: 87 × 3
person height mass_kg
<chr> <int> <dbl>
1 Luke Skywalker 172 77
2 C-3PO 167 75
3 R2-D2 96 32
4 Darth Vader 202 136
5 Leia Organa 150 49
6 Owen Lars 178 120
7 Beru Whitesun Lars 165 75
8 R5-D4 97 32
9 Biggs Darklighter 183 84
10 Obi-Wan Kenobi 182 77
# ℹ 77 more rows
4.6.5.1 where()
where()
is a useful helper function that comes in handy, for example, when we want to select columns based on their data type.
# A tibble: 87 × 3
height mass birth_year
<int> <dbl> <dbl>
1 172 77 19
2 167 75 112
3 96 32 33
4 202 136 41.9
5 150 49 19
6 178 120 52
7 165 75 47
8 97 32 NA
9 183 84 24
10 182 77 57
# ℹ 77 more rows
The following selects all columns that are not numeric:
# A tibble: 87 × 11
name hair_color skin_color eye_color sex gender homeworld species films
<chr> <chr> <chr> <chr> <chr> <chr> <chr> <chr> <lis>
1 Luke Sk… blond fair blue male mascu… Tatooine Human <chr>
2 C-3PO <NA> gold yellow none mascu… Tatooine Droid <chr>
3 R2-D2 <NA> white, bl… red none mascu… Naboo Droid <chr>
4 Darth V… none white yellow male mascu… Tatooine Human <chr>
5 Leia Or… brown light brown fema… femin… Alderaan Human <chr>
6 Owen La… brown, gr… light blue male mascu… Tatooine Human <chr>
7 Beru Wh… brown light blue fema… femin… Tatooine Human <chr>
8 R5-D4 <NA> white, red red none mascu… Tatooine Droid <chr>
9 Biggs D… black light brown male mascu… Tatooine Human <chr>
10 Obi-Wan… auburn, w… fair blue-gray male mascu… Stewjon Human <chr>
# ℹ 77 more rows
# ℹ 2 more variables: vehicles <list>, starships <list>
Note that I used ~
here to indicate that I’m creating an anonymous function to check whether column type is numeric. A one-sided formula (expression beginning with ~
) is interpreted as function(x)
, and wherever x
would go in the function is represented by .
.
So, I could write the same code like so:
df.starwars %>%
select(where(function(x) !is.numeric(x))) # selects all columns that are not numeric
# A tibble: 87 × 11
name hair_color skin_color eye_color sex gender homeworld species films
<chr> <chr> <chr> <chr> <chr> <chr> <chr> <chr> <lis>
1 Luke Sk… blond fair blue male mascu… Tatooine Human <chr>
2 C-3PO <NA> gold yellow none mascu… Tatooine Droid <chr>
3 R2-D2 <NA> white, bl… red none mascu… Naboo Droid <chr>
4 Darth V… none white yellow male mascu… Tatooine Human <chr>
5 Leia Or… brown light brown fema… femin… Alderaan Human <chr>
6 Owen La… brown, gr… light blue male mascu… Tatooine Human <chr>
7 Beru Wh… brown light blue fema… femin… Tatooine Human <chr>
8 R5-D4 <NA> white, red red none mascu… Tatooine Droid <chr>
9 Biggs D… black light brown male mascu… Tatooine Human <chr>
10 Obi-Wan… auburn, w… fair blue-gray male mascu… Stewjon Human <chr>
# ℹ 77 more rows
# ℹ 2 more variables: vehicles <list>, starships <list>
For more details, take a look at the help file for select()
, and this this great tutorial in which I learned about some of the more advanced ways of using select()
.
4.6.6 Practice 2
Create a data frame that:
- only has the species Human
and Droid
- with the following data columns (in this order): name, species, birth_year, homeworld
- is arranged according to birth year (with the lowest entry at the top of the data frame)
- and has the name
column renamed to person
4.6.7 mutate()
mutate()
is used to change existing columns or make new ones.
df.starwars %>%
mutate(height = height / 100, # to get height in meters
bmi = mass / (height^2)) %>% # bmi = kg / (m^2)
select(name, height, mass, bmi)
# A tibble: 87 × 4
name height mass bmi
<chr> <dbl> <dbl> <dbl>
1 Luke Skywalker 1.72 77 26.0
2 C-3PO 1.67 75 26.9
3 R2-D2 0.96 32 34.7
4 Darth Vader 2.02 136 33.3
5 Leia Organa 1.5 49 21.8
6 Owen Lars 1.78 120 37.9
7 Beru Whitesun Lars 1.65 75 27.5
8 R5-D4 0.97 32 34.0
9 Biggs Darklighter 1.83 84 25.1
10 Obi-Wan Kenobi 1.82 77 23.2
# ℹ 77 more rows
Here, I’ve calculated the bmi for the different starwars characters. I first mutated the height variable by going from cm to m, and then created the new column “bmi”.
A useful helper function for mutate()
is ifelse()
which is a shorthand for the if-else control flow (Section 4.3.4.1). Here is an example:
df.starwars %>%
mutate(height_categorical = ifelse(test = height > median(height, na.rm = T),
yes = "tall",
no = "short")) %>%
select(name, contains("height"))
# A tibble: 87 × 3
name height height_categorical
<chr> <int> <chr>
1 Luke Skywalker 172 short
2 C-3PO 167 short
3 R2-D2 96 short
4 Darth Vader 202 tall
5 Leia Organa 150 short
6 Owen Lars 178 short
7 Beru Whitesun Lars 165 short
8 R5-D4 97 short
9 Biggs Darklighter 183 tall
10 Obi-Wan Kenobi 182 tall
# ℹ 77 more rows
ifelse()
works in the following way: we first specify the condition, then what should be returned if the condition is true, and finally what should be returned otherwise. The more verbose version of the statement above would be: ifelse(test = height > median(height, na.rm = T), yes = "tall", no = "short")
In previous versions of dplyr
(the package we use for data wrangling), there were a variety of additional mutate functions such as mutate_at()
, mutate_if()
, and mutate_all()
. In the most recent version of dplyr
, these additional functions have been deprecated, and replaced with the flexible across()
helper function.
4.6.7.1 across()
across()
allows us to use the syntax that we’ve learned for select()
to select particular variables and apply a function to each of the selected variables.
For example, let’s imagine that we want to z-score a number of variables in our data frame. We can do this like so:
# A tibble: 87 × 14
name height[,1] mass[,1] hair_color skin_color eye_color birth_year[,1]
<chr> <dbl> <dbl> <chr> <chr> <chr> <dbl>
1 Luke Skyw… -0.0749 -0.120 blond fair blue -0.443
2 C-3PO -0.219 -0.132 <NA> gold yellow 0.158
3 R2-D2 -2.26 -0.385 <NA> white, bl… red -0.353
4 Darth Vad… 0.788 0.228 none white yellow -0.295
5 Leia Orga… -0.708 -0.285 brown light brown -0.443
6 Owen Lars 0.0976 0.134 brown, gr… light blue -0.230
7 Beru Whit… -0.276 -0.132 brown light blue -0.262
8 R5-D4 -2.23 -0.385 <NA> white, red red NA
9 Biggs Dar… 0.241 -0.0786 black light brown -0.411
10 Obi-Wan K… 0.213 -0.120 auburn, w… fair blue-gray -0.198
# ℹ 77 more rows
# ℹ 7 more variables: sex <chr>, gender <chr>, homeworld <chr>, species <chr>,
# films <list>, vehicles <list>, starships <list>
In the .cols =
argument of across()
, I’ve specified what variables to mutate. In the .fns =
argument, I’ve specified that I want to use the function scale
. Note that I wrote the function without ()
. The .fns
argument expects allows these possible values:
- the function itself, e.g.
mean
- a call to the function with
.
as a dummy argument,~ mean(.)
(note the~
before the function call) - a list of functions
list(mean = mean, median = ~ median(.))
(where I’ve mixed both of the other ways)
We can also use names to create new columns:
df.starwars %>%
mutate(across(.cols = c(height, mass, birth_year),
.fns = scale,
.names = "{.col}_z")) %>%
select(name, contains("height"), contains("mass"), contains("birth_year"))
# A tibble: 87 × 7
name height height_z[,1] mass mass_z[,1] birth_year birth_year_z[,1]
<chr> <int> <dbl> <dbl> <dbl> <dbl> <dbl>
1 Luke Skywal… 172 -0.0749 77 -0.120 19 -0.443
2 C-3PO 167 -0.219 75 -0.132 112 0.158
3 R2-D2 96 -2.26 32 -0.385 33 -0.353
4 Darth Vader 202 0.788 136 0.228 41.9 -0.295
5 Leia Organa 150 -0.708 49 -0.285 19 -0.443
6 Owen Lars 178 0.0976 120 0.134 52 -0.230
7 Beru Whites… 165 -0.276 75 -0.132 47 -0.262
8 R5-D4 97 -2.23 32 -0.385 NA NA
9 Biggs Darkl… 183 0.241 84 -0.0786 24 -0.411
10 Obi-Wan Ken… 182 0.213 77 -0.120 57 -0.198
# ℹ 77 more rows
I’ve specified how I’d like the new variables to be called by using the .names =
argument of across()
. {.col}
stands of the name of the original column, and here I’ve just added _z
to each column name for the scaled columns.
We can also apply several functions at the same time.
df.starwars %>%
mutate(across(.cols = c(height, mass, birth_year),
.fns = list(z = scale,
centered = ~ scale(., scale = FALSE)))) %>%
select(name, contains("height"), contains("mass"), contains("birth_year"))
# A tibble: 87 × 10
name height height_z[,1] height_centered[,1] mass mass_z[,1]
<chr> <int> <dbl> <dbl> <dbl> <dbl>
1 Luke Skywalker 172 -0.0749 -2.60 77 -0.120
2 C-3PO 167 -0.219 -7.60 75 -0.132
3 R2-D2 96 -2.26 -78.6 32 -0.385
4 Darth Vader 202 0.788 27.4 136 0.228
5 Leia Organa 150 -0.708 -24.6 49 -0.285
6 Owen Lars 178 0.0976 3.40 120 0.134
7 Beru Whitesun Lars 165 -0.276 -9.60 75 -0.132
8 R5-D4 97 -2.23 -77.6 32 -0.385
9 Biggs Darklighter 183 0.241 8.40 84 -0.0786
10 Obi-Wan Kenobi 182 0.213 7.40 77 -0.120
# ℹ 77 more rows
# ℹ 4 more variables: mass_centered <dbl[,1]>, birth_year <dbl>,
# birth_year_z <dbl[,1]>, birth_year_centered <dbl[,1]>
Here, I’ve created z-scored and centered (i.e. only subtracted the mean but didn’t divide by the standard deviation) versions of the height
, mass
, and birth_year
columns in one go.
You can use the everything()
helper function if you want to apply a function to all of the columns in your data frame.
df.starwars %>%
select(height, mass) %>%
mutate(across(.cols = everything(),
.fns = as.character)) # transform all columns to characters
# A tibble: 87 × 2
height mass
<chr> <chr>
1 172 77
2 167 75
3 96 32
4 202 136
5 150 49
6 178 120
7 165 75
8 97 32
9 183 84
10 182 77
# ℹ 77 more rows
Here, I’ve selected some columns first, and then changed the mode to character in each of them.
Sometimes, you want to apply a function only to those columns that have a particular data type. This is where where()
comes in handy!
For example, the following code changes all the numeric columns to character columns:
# A tibble: 87 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <chr> <chr> <chr> <chr> <chr> <chr> <chr> <chr>
1 Luke Sk… 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth V… 202 136 none white yellow 41.9 male mascu…
5 Leia Or… 150 49 brown light brown 19 fema… femin…
6 Owen La… 178 120 brown, gr… light blue 52 male mascu…
7 Beru Wh… 165 75 brown light blue 47 fema… femin…
8 R5-D4 97 32 <NA> white, red red <NA> none mascu…
9 Biggs D… 183 84 black light brown 24 male mascu…
10 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
# ℹ 77 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
Or we could round all the numeric columns to one digit:
# A tibble: 87 × 14
name height mass hair_color skin_color eye_color birth_year sex gender
<chr> <dbl> <dbl> <chr> <chr> <chr> <dbl> <chr> <chr>
1 Luke Sk… 172 77 blond fair blue 19 male mascu…
2 C-3PO 167 75 <NA> gold yellow 112 none mascu…
3 R2-D2 96 32 <NA> white, bl… red 33 none mascu…
4 Darth V… 202 136 none white yellow 41.9 male mascu…
5 Leia Or… 150 49 brown light brown 19 fema… femin…
6 Owen La… 178 120 brown, gr… light blue 52 male mascu…
7 Beru Wh… 165 75 brown light blue 47 fema… femin…
8 R5-D4 97 32 <NA> white, red red NA none mascu…
9 Biggs D… 183 84 black light brown 24 male mascu…
10 Obi-Wan… 182 77 auburn, w… fair blue-gray 57 male mascu…
# ℹ 77 more rows
# ℹ 5 more variables: homeworld <chr>, species <chr>, films <list>,
# vehicles <list>, starships <list>
4.7 Additional resources
4.7.1 Cheatsheets
- base R –> summary of how to use base R (we will mostly use the tidyverse but it’s still important to know how to do things in base R)
- data transformation –> transforming data using
dplyr
4.8 Session info
Information about this R session including which version of R was used, and what packages were loaded.
R version 4.4.2 (2024-10-31)
Platform: aarch64-apple-darwin20
Running under: macOS Sequoia 15.2
Matrix products: default
BLAS: /Library/Frameworks/R.framework/Versions/4.4-arm64/Resources/lib/libRblas.0.dylib
LAPACK: /Library/Frameworks/R.framework/Versions/4.4-arm64/Resources/lib/libRlapack.dylib; LAPACK version 3.12.0
locale:
[1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8
time zone: America/Los_Angeles
tzcode source: internal
attached base packages:
[1] stats graphics grDevices utils datasets methods base
other attached packages:
[1] lubridate_1.9.3 forcats_1.0.0 stringr_1.5.1 dplyr_1.1.4
[5] purrr_1.0.2 readr_2.1.5 tidyr_1.3.1 tibble_3.2.1
[9] ggplot2_3.5.1 tidyverse_2.0.0 DT_0.33 visdat_0.6.0
[13] skimr_2.1.5 knitr_1.49
loaded via a namespace (and not attached):
[1] sass_0.4.9 utf8_1.2.4 generics_0.1.3 stringi_1.8.4
[5] hms_1.1.3 digest_0.6.36 magrittr_2.0.3 evaluate_0.24.0
[9] grid_4.4.2 timechange_0.3.0 bookdown_0.42 fastmap_1.2.0
[13] jsonlite_1.8.8 fansi_1.0.6 crosstalk_1.2.1 scales_1.3.0
[17] jquerylib_0.1.4 cli_3.6.3 rlang_1.1.4 munsell_0.5.1
[21] base64enc_0.1-3 withr_3.0.2 repr_1.1.7 cachem_1.1.0
[25] yaml_2.3.10 tools_4.4.2 tzdb_0.4.0 colorspace_2.1-0
[29] vctrs_0.6.5 R6_2.5.1 lifecycle_1.0.4 htmlwidgets_1.6.4
[33] pkgconfig_2.0.3 pillar_1.9.0 bslib_0.7.0 gtable_0.3.5
[37] glue_1.8.0 xfun_0.49 tidyselect_1.2.1 farver_2.1.2
[41] htmltools_0.5.8.1 labeling_0.4.3 rmarkdown_2.29 compiler_4.4.2