-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy path03-functional-programming.qmd
337 lines (259 loc) · 11.3 KB
/
03-functional-programming.qmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
---
filters:
- webr
webr:
packages: ['dplyr', 'purrr', 'tidyr', 'withr']
---
# A primer on functional programming
<div style="text-align:center;">
```{r, echo = F}
knitr::include_graphics("img/lambda.png")
```
</div>
What you'll have learned by the end of the chapter: writing your own functions,
functional programming basics (map, reduce, anonymous functions and higher-order
functions).
## Introduction
Functional programming is a way of writing programs that relies exclusively on
the evalutation of functions. Mathematical functions have a very neat property:
for any given input, they ALWAYS return exactly the same output. This is what we
want to achieve with the functions that we will write. Functions that always
return the same result are called pure, and a language that only allows writing
pure functions is called a pure functional programming language. R is not a pure
functional programming language, so we have to be careful not to write impure
functions that manipulate the global state.
But what is state? Run the following code in your console:
```{webr-r}
ls()
```
This will list every object defined in the global environment. Now run the
following line:
```{webr-r}
x <- 1
```
and then `ls()` again. `x` should now be listed alongside the other objects. You
just manipulated the state of your current R session. Now if you run something
like:
```{webr-r}
x + 1
```
This will produce `2`. We want to avoid pipelines that depend on some definition
of some global variable somewhere, which could be subject to change, because
this could mean that 2 different runs of the same pipeline could produce 2
different results. Notice that I used the verb *avoid* in the sentence before.
This is sometimes not possible to avoid. Such situations have to be carefully
documented and controlled.
As a more realistic example, imagine that within the pipeline you set up, some
random numbers are generated. For example, to generate 10 random draws from a
normal distribution:
```{webr-r}
rnorm(n = 10)
```
Each time you run this line, you will get another set of 10 random numbers. This
is obviously a good thing in interactive data analysis, but much less so when
running a pipeline programmatically. R provides a way to fix the random seed,
which will make sure you always get the same random numbers:
```{webr-r}
set.seed(1234)
rnorm(n = 10)
```
But `set.seed()` only works for one call, so you must call it again if you need
the random numbers again:
```{webr-r}
set.seed(1234)
rnorm(10)
```
```{webr-r}
rnorm(10)
```
```{webr-r}
set.seed(1234)
rnorm(10)
```
The problem with `set.seed()` is that you only partially solve the problem of
`rnorm()` not being pure; this is because while `rnorm()` now does return the
same output for the same input, this only works if you manipulate the state of
your program to change the seed beforehand. Ideally, we would like to have a
pure version of `rnorm()`, which would be self-contained and not depend on the
value of the seed defined in the global environment. There is a package
developped by Posit (the makers of RStudio and the packages from the
*tidyverse*), called `{withr}` which allows to rewrite our functions in a pure
way. `{withr}` has several functions, all starting with `with_` that allow users
to run code with some temporary defined variables, without altering the global
environment. For example, it is possible to run a `rnorm()` with a seed, using
`withr::with_seed()`:
```{webr-r}
library(withr)
with_seed(seed = 1234, {
rnorm(10)
})
```
But ideally you’d want to go a step further and define a new function
that is pure. To turn an impure function into a pure function, you usually only
need to add some arguments to it. This is how we would create a `pure_rnorm()`
function:
```{webr-r}
pure_rnorm <- function(..., seed){
with_seed(seed, rnorm(...))
}
pure_rnorm(10, seed = 1234)
```
`pure_rnorm()` is now self-contained, and does not pollute the global
environment. We’re going to learn how to write functions in just a
bit, so don’t worry if the code above does not make sense yet.
<div style="text-align:center;">
```{r, echo = F}
knitr::include_graphics("img/cat_loops.png")
```
</div>
A very practical consequence of using functional programming is that loops are
not used, because loops are imperative and imperative programming is all about
manipulating state. However, there are situations where loops are more efficient
than the alternative (in R at least). So we will still learn and use them, but
only when absolutely necessary, and we will always encapsulate a loop inside a
function. Just like with the example above, this ensures that we have a pure,
self-contained function that we can reason about easily. What I mean by this, is
that loops are not always very easy to decipher. The concept of loops is simple
enough: take this instruction, and repeat it N times. But in practice, if
you’re reading code, it is not possible to understand what a loop is
doing at first glance. There are only two solutions in this case:
- you’re lucky and there are comments that explain what the loop is doing;
- you have to let the loop run either in your head or in a console with some examples to really understand whit is going on.
For example, consider the following code:
```{webr-r}
library(dplyr)
data(starwars)
sum_humans <- 0
sum_others <- 0
n_humans <- 0
n_others <- 0
for(i in seq_along(1:nrow(starwars))){
if(!is.na(unlist(starwars[i, "species"])) &
unlist(starwars[i, "species"]) == "Human"){
if(!is.na(unlist(starwars[i, "height"]))){
sum_humans <- sum_humans + unlist(starwars[i, "height"])
n_humans <- n_humans + 1
} else {
0
}
} else {
if(!is.na(unlist(starwars[i, "height"]))){
sum_others <- sum_others + unlist(starwars[i, "height"])
n_others <- n_others + 1
} else {
0
}
}
}
mean_height_humans <- sum_humans/n_humans
mean_height_others <- sum_others/n_others
```
What this does is not immediately obvious. The only hint you get are the two
last lines, where you can read that we compute the average height for humans and
non-humans in the sample. And this code could look a lot worse, because I am
using functions like `is.na()` to test if a value is `NA` or not, and
I’m using `unlist()` as well. If you compare this mess to a
functional approach, I hope that I can stop my diatribe against imperative style
programming here:
```{webr-r}
starwars %>%
group_by(is_human = species == "Human") %>%
summarise(mean_height = mean(height, na.rm = TRUE))
```
Not only is this shorter, it doesn’t even need any comments to
explain what’s going on. If you’re using functions with
explicit names, the code becomes self-explanatory.
The other advantage of a functional (also called declarative) programming style
is that you get function composition for free. Function composition is an
operation that takes two functions *g* and *f* and returns a new function *h*
such that $h(x) = g(f(x))$. Formally:
```
h = g ∘ f such that h(x) = g(f(x))
```
`∘` is the composition operator. You can read `g ∘ f` as
*g after f*. When using functional programming, you can compose functions very
easily, simply by using `|>` or `%>%`:
```{r, eval = F}
h <- f |> g
```
`f |> g` can be read as *f then g*, which is equivalent to *g after f*. Function
composition might not seem like a big deal, but it actually is. If we structure
our programs in this way, as a sequence of function calls, we get many benefits.
Functions are easy to test, document, maintain, share and can be composed. This
allows us to very succintly express complex workflows:
```{webr-r}
starwars %>%
filter(skin_color == "light") %>%
select(species, sex, mass) %>%
group_by(sex, species) %>%
summarise(
total_individuals = n(),
min_mass = min(mass, na.rm = TRUE),
mean_mass = mean(mass, na.rm = TRUE),
sd_mass = sd(mass, na.rm = TRUE),
max_mass = max(mass, na.rm = TRUE),
.groups = "drop"
) %>%
select(-species) %>%
tidyr::pivot_longer(-sex, names_to = "statistic", values_to = "value")
```
Needless to say, writing this in an imperative approach would be quite
complicated.
Another consequence of using functional programming is that our code will live
in plain text files, and not in Jupyter (or equivalent) notebooks. Not only does
imperative code have state, but notebooks themselves have a (hidden) state. You
should avoid notebooks at all costs, even for experimenting.
## Defining your own functions
Let's first learn about actually writing functions. Read [chapter
7](https://b-rodrigues.github.io/modern_R/defining-your-own-functions.html) of
my other book.
The most important concepts for this course are discussed in the following
sections:
- functions that take functions as arguments [(section 7.4)](https://b-rodrigues.github.io/modern_R/defining-your-own-functions.html#functions-that-take-functions-as-arguments-writing-your-own-higher-order-functions)
```{webr-r}
my_func <- function(x, func, ...){
func(x, ...)
}
my_func(c(1, 8, 1, NA, 8), mean, na.rm = TRUE)
```
- functions that take data (and the data's columns) as arguments [(section 7.6)](https://b-rodrigues.github.io/modern_R/defining-your-own-functions.html#functions-that-take-columns-of-data-as-arguments);
```{webr-r}
simple_function <- function(dataset, col_name, var_name, fn_name){
dataset %>%
group_by(across({{col_name}})) %>%
summarise("{{fn_name}}_{{var_name}}" := {{fn_name}}({{var_name}}))
}
# Group by one column
simple_function(mtcars, vs, mpg, sd)
# Group by several columns
simple_function(mtcars, c(am, vs), mpg, sd)
# Even more general: group by any columns and apply any number of functions
simple_function <- function(dataset, cols, vars, fns){
dataset %>%
group_by(across({{cols}})) %>%
summarise(across({{vars}}, fns, .names = "{.fn}_{.col}"))
}
simple_function(mtcars, c(am, vs), c(mpg, hp), list("mean" = mean, "sd" = sd))
```
Read more about it [here](https://dplyr.tidyverse.org/reference/across.html) and
[here](https://dplyr.tidyverse.org/articles/programming.html).
## Functional programming
You should ideally work through the whole of chapter 7, and then tackle [chapter
8](https://b-rodrigues.github.io/modern_R/functional-programming.html). What's
important there are:
- `purrr::map()`, `purrr::reduce()` (sections [8.3.1](https://b-rodrigues.github.io/modern_R/functional-programming.html#doing-away-with-loops-the-map-family-of-functions) and [8.3.2](https://b-rodrigues.github.io/modern_R/functional-programming.html#reducing-with-purrr))
Apply a function to each element of a vector or list:
```{webr-r}
map(seq(1:10), sqrt)
```
(`lapply()` is a base function that works similarly to `purrr::map()`)
Reduce a vector to a single element by iteratively applying a function:
```{webr-r}
reduce(seq(1:10), `+`)
```
- And list based workflows (section [8.4](https://b-rodrigues.github.io/modern_R/functional-programming.html#functional-programming-and-plotting))
## Further reading
- [Cleaner R Code with Functional Programming](https://towardsdatascience.com/cleaner-r-code-with-functional-programming-adc37931ef7a)
- [Functional Programming (Chapter from Advanced R)](https://adv-r.hadley.nz/fp.html)
- [Why you should(n't) care about Monads if you're an R programmer](https://www.brodrigues.co/blog/2022-04-11-monads/)
- [Some learnings from functional programming you can use to write safer programs](https://www.brodrigues.co/blog/2022-05-26-safer_programs/)