-
Notifications
You must be signed in to change notification settings - Fork 41
/
formules.Rmd
313 lines (215 loc) · 14.6 KB
/
formules.Rmd
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
---
title: "Formules"
---
```{r options_communes, include = FALSE}
source("options_communes.R")
```
Ce chapitre vise à illustrer l'utilisation de la <dfn>notation <q>formule</q></dfn> de **R**, qui désigne l'emploi de cette notation par l'expression `formula`{data-pkg="stats"}. Cette notation est utilisée par de très nombreuses fonctions de **R** : on en a notamment vu plusieurs exemples dans le [chapitre sur les graphiques bivariés](graphiques-bivaries.html), car l'extension `ggplot2`{.pkg} se sert de cette notation dans ses paramètres `facet_wrap`{data-pkg="ggplot2"} et `facet_grid`{data-pkg="ggplot2"}.
Dans ce chapitre, on verra comment se servir de la notation <q>formule</q> dans deux contextes différents. D'une part, on verra que deux fonctions basiques de **R** se servent de cette notation pour produire des tableaux croisés et des statistiques bivariées. D'autre part, on verra que l'extension `lattice`{.pkg} se sert de cette notation pour créer des graphiques <q>panelisés</q>, dits graphiques à <q>petits multiples</q>.
Dans plusieurs autres chapitres, les opérations décrites ci-dessus sont effectuées avec les extensions `dplyr`{.pkg} d'une part, et `ggplot2`{.pkg} d'autre part. On se servira également de ces extensions dans ce chapitre, de manière à mener une comparaison des différentes manières d'effectuer certaines opérations dans **R**, avec ou sans la notation <q>formule</q> :
```{r, message = FALSE}
library(dplyr)
library(ggplot2)
```
## Statistiques descriptives
Les premiers exemples de ce chapitre montrent l'utilisation de cette notation pour produire des tableaux croisés et des statistiques descriptives. Le jeu de données utilisé, `hdv2003`{data-pkg="questionr"}, a déjà été utilisé dans plusieurs chapitres, et font partie de l'extension `questionr`{.pkg}. Chargeons cette extension et le jeu de données `hdv2003`{data-pkg="questionr"} :
```{r}
library(questionr)
data(hdv2003)
```
Pour rappel, ce jeu de données contient des individus, leur âge, leur statut professionnel, et le nombre d'heures quotidiennes passées à regarder la télévision.
```{r}
glimpse(hdv2003, 75)
```
### Tableaux croisés avec xtabs
Utilisons, pour ce premier exemple, la variable <var>occup</var> du jeu de données `hdv2003`{data-pkg="questionr"}, qui correspond au statut professionnel des individus inclus dans l'échantillon. La fonction de base pour compter les individus par statut est la fonction `table`{data-pkg="base"} :
```{r}
table(hdv2003$occup)
```
Avec la fonction `xtabs`{data-pkg="stats"}, le même résultat est produit à partir de la notation suivante :
```{r}
xtabs(~ occup, data = hdv2003)
```
Le premier argument est une formule, au sens où **R** entend cette expression. Le second argument, `data`, correspond au jeu de données auquel la formule doit être appliquée. On pourra se passer d'écrire explicitement cet argument dans les exemples suivants.
L'avantage de la fonction `xtabs`{data-pkg="stats"} n'est pas évident dans ce premier exemple. En réalité, cette fonction devient utile lorsque l'on souhaite construire un ou plusieurs tableau(x) croisé(s). Par exemple, pour croiser la variable <var>occup</var> avec la variable <var>sexe</var>, une solution constiste à écrire :
```{r}
with(hdv2003, table(occup, sexe))
```
Ou alors, ce qui revient au même :
```{r, eval = FALSE}
table(hdv2003$occup, hdv2003$sexe)
```
Avec `xtabs`{data-pkg="stats"}, la même opération s'écrit de la manière suivante :
```{r}
xtabs(~ occup + sexe, hdv2003)
```
Cette écriture est plus courte que le code équivalent dans `dplyr`{.pkg} :
```{r}
hdv2003 %>%
group_by(occup) %>%
summarise(Homme = sum(sexe == "Homme"),
Femme = sum(sexe == "Femme"))
```
Par contre, on pourra éventuellement utiliser `count`{data-pkg="dplyr"} de `dplyr`{.pkg}. ATTENTION : le format du résultat ne sera pas le même.
```{r}
hdv2003 %>%
group_by(occup) %>%
dplyr::count(sexe)
```
Pour un tableau croisé joliment mis en forme, on pourra avoir recours à `tbl_cross`{data-pkg="gtsummary"} de `gtsummary`{.pkg}.
```{r}
library(gtsummary)
hdv2003 %>%
tbl_cross(row = "occup", col = "sexe", percent = "row")
```
De plus, `xtabs`{data-pkg="stats"} permet de créer plusieurs tableaux croisés en une seule formule :
```{r}
xtabs(~ occup + sexe + trav.imp, hdv2003)
```
Cet exemple permet simplement de réaliser que la variable <var>trav.imp</var>, qui contient les réponses à une question portant sur l'importance du travail, n'a été mesurée (c'est-à-dire que la question n'a été posée) qu'aux seuls individus actifs de l'échantillon.
### Statistiques bivariées avec aggregate
```{r}
aggregate(heures.tv ~ sexe, mean, data = hdv2003)
```
Ici, le premier argument est à nouveau une formule. Le second argument correspond à la statistique descriptive que l'on souhaite obtenir, et le dernier argument indique le jeu de données auquel appliquer les deux autres arguments. On peut d'ailleurs obtenir le même résultat en respectant de manière plus stricte l'ordre des arguments dans la syntaxe de la fonction `aggregate`{data-pkg="stats"} :
```{r}
aggregate(heures.tv ~ sexe, hdv2003, mean)
```
Cette écriture est, à nouveau, plus compacte que le code équivalent dans `dplyr`{.pkg}, qui demande de spécifier le retrait des valeurs manquantes :
```{r, eval = FALSE}
hdv2003 %>
group_by(sexe) %>%
summarise(heures.tv = mean(heures.tv, na.rm = TRUE))
```
À nouveau, on va pouvoir combiner plusieurs variables dans la formule que l'on passe à `aggregate`{data-pkg="stats"}, ce qui va permettre d'obtenir la moyenne des heures de télévision quotidiennes par sexe et par statut professionnel :
```{r}
aggregate(heures.tv ~ sexe + occup, hdv2003, mean)
```
La même opération `dplyr`{.pkg} :
```{r, eval = FALSE}
hdv2003 %>% group_by(occup, sexe) %>%
summarise(heures.tv = mean(heures.tv, na.rm = TRUE))
```
La fonction `aggregate`{data-pkg="stats"} permet bien sûr d'utiliser une autre fonction que la moyenne, comme dans cet exemple, suivi de son équivalent avec `dplyr`{.pkg} :
```{r}
# âge médian par sexe et statut professionnel
aggregate(age ~ sexe + occup, hdv2003, median)
```
```{r, eval = FALSE}
# code équivalent avec l'extension 'dplyr'
hdv2003 %>%
group_by(occup, sexe) %>%
summarise(age = median(age, na.rm = TRUE))
```
Si, comme dans le cas de `summarise`{data-pkg="dplyr"}, on souhaite passer des arguments supplémentaires à la fonction `median`{data-pkg="stats"}, il suffit de les lister à la suite du nom de la fonction. Par exemple, on écrirait : `aggregate(age ~ sexe + occup, hdv2003, median, na.rm = TRUE)`. Ceci étant, `aggregate`{data-pkg="stats"} utilise par défaut l'option `na.action = na.omit`, donc il est bon de se rappeler que l'on peut désactiver cette option en utilisant l'option `na.action = na.pass`, ce qui permet éventuellement de conserver des lignes vides dans le tableau de résultat.
La fonction `aggregate`{data-pkg="stats"} permet, par ailleurs, d'obtenir des résultats à plusieurs colonnes. Dans l'exemple ci-dessus, on illustre ce principe avec la fonction `range`{data-pkg="stats"}, qui renvoie deux résultats (la valeur minimale et la valeur maximale de la variable, qui est toujours la variable <var>age</var>), chacun présentés dans une colonne :
```{r, paged.print = FALSE}
aggregate(age ~ sexe + occup, hdv2003, range)
```
Cette fonction ne peut pas être facilement écrite dans `dplyr`{.pkg} sans réécrire chacune des colonnes, ce que le bloc de code suivant illustre. On y gagne en lisibilité dans les intitulés de colonnes :
```{r, eval = FALSE}
hdv2003 %>%
group_by(occup, sexe) %>%
summarise(min = min(age, na.rm = TRUE),
max = max(age, na.rm = TRUE))
```
Depuis la version 1.0.0 de `dplyr`{.pkg}, `summarise`{data-pkg="dplyr"} accepte maintenant des fonctions pouvant renvoyer plusieurs valeurs, créant ainsi autant de lignes (voir <https://www.tidyverse.org/blog/2020/03/dplyr-1-0-0-summarise/>).
```{r}
hdv2003 %>%
group_by(occup, sexe) %>%
summarise(age = range(age, na.rm = TRUE), type = c("min", "max"))
```
On pourrait de même définir sa propre fonction et la passer à `aggregate`{data-pkg="stats"} :
```{r, paged.print = FALSE}
f <- function(x) c(mean = mean(x, na.rm = TRUE), sd = sd(x, na.rm = TRUE))
aggregate(age ~ sexe + occup, hdv2003, f)
```
Mais on réalisera vite une des limitations de `aggregate`{data-pkg="stats"} dans ce cas-là : le tableau retourné ne contient pas 4 colonnes, mais 3 uniquement, ce que l'on peut vérifier à l'aide de `dim`{data-pkg="base"} ou `str`{data-pkg="utils"}.
```{r}
str(aggregate(age ~ sexe + occup, hdv2003, f))
```
Pour ce type d'opération, dans lequel on souhaite récupérer plusieurs variables calculées afin de travailler sur ces données agrégées soit dans le cadre d'opérations numériques soit de constructions graphiques, `dplyr`{.pkg} ou `Hmisc`{.pkg} s'avèrent plus commodes. Voici un exemple avec `summarize`{data-pkg="Hmisc"} de l'extension `Hmisc`{.pkg} :
```{r}
library(Hmisc, quietly = TRUE)
with(hdv2003, summarize(age, llist(sexe, occup), f))
```
Notons que `Hmisc`{.pkg} offre déjà une telle fonction (`smean.sd`{data-pkg="Hmisc"}), ce qui nous aurait épargné d'écrire notre propre fonction, `f`, et il en existe bien d'autres. Voici un exemple avec des intervalles de confiance estimés par bootstrap :
```{r}
with(hdv2003, summarize(age, llist(sexe, occup), smean.cl.boot))
```
Et un exemple avec `dplyr`{.pkg}.
```{r}
hdv2003 %>%
group_by(sexe, occup) %>%
summarise(
tibble(
age_mean = mean(age, na.rm = TRUE),
age_sd = sd(age, na.rm = TRUE)
)
)
```
Enfin, il est également possible d'utiliser plusieurs variables numériques à gauche de l'opérateur `~`. En voici une illustration :
```{r}
aggregate(cbind(age,poids) ~ sexe + occup, hdv2003, mean)
```
## Panels graphiques avec lattice
Les exemples suivants montreront ensuite comment la notation <q>formule</q> peut servir à produire des graphiques par panel avec l'extension `lattice`{.pkg}.
```{r}
library(lattice)
```
<div class="note">
L'extension `lattice`{.pkg} présente l'avantage d'être installée par défaut avec **R**. Il n'est donc pas nécessaire de l'installer préalablement.
</div>
Chargeons les mêmes données que le [chapitre sur les graphiques bivariés](graphiques-bivaries.html).
```{r}
# charger l'extension lisant le format CSV
library(readr)
# emplacement souhaité pour le jeu de données
file = "data/debt.csv"
# télécharger le jeu de données s'il n'existe pas
if(!file.exists(file))
download.file("http://www.stat.cmu.edu/~cshalizi/uADA/13/hw/11/debt.csv",
file, mode = "wb")
# charger les données dans l'objet 'debt'
debt = read_csv(file)
```
Rejetons rapidement un coup d'oeil à ces données, qui sont structurées par pays (variable <var>Country</var>) et par année (variable <var>Year</var>). On y trouve deux variables, <var>growth</var> (le taux de croissance du produit intérieur brut réel), et <var>ratio</var> (le ratio entre la dette publique et le produit intérieur brut), ainsi qu'une première colonne vide, ne contenant que des numéros lignes, dont on va se débarrasser :
```{r}
# inspection des données
glimpse(debt, 75)
# suppression de la première colonne
debt = debt[, -1 ]
```
### Visualisation bivariée
Le même graphique s'écrit de la manière suivante avec l'extension `lattice`{.pkg} :
```{r}
xyplot(growth ~ Year, data = debt)
```
### Visualisation par <q>petits multiples</q>
Appliquons désormais la même visualisation par <q>petits multiples</q> que vue dans le chapitre :
```{r}
xyplot(growth ~ Year | Country, data = debt)
```
Enfin, rajoutons quelques options au graphique, afin de montrer comment l'extension `lattice`{.pkg} fonctionne :
```{r}
xyplot(growth ~ Year | Country,
type = c("o", "l"),
main = "Données Reinhart et Rogoff corrigées, 1946-2009",
ylab = "Taux de croissance du PIB",
xlab = NULL,
data = debt)
```
## Spécifier des modèles
### Les formules R
En réalité, la notation par formule qu'utilise **R** est celle proposée par Wilkinson *et al.* dans les années 70 pour schématiser la relation entre plusieurs variables dans un plan d'expérience. Plus spécifiquement, l'idée revient à exprimer une <dfn>relation <q>fonctionnelle</q></dfn>, symbolisée par l'opérateur `~`, entre une variable réponse `y` et une ou plusieurs variables explicatives. Disons, pour simplifier, que `y` est une variable d'intérêt (numérique ou facteur selon le type de modèle), `x` une variable numérique et que `a` et `b` sont des variables catégorielles (des facteurs dans le langage **R**). Voici les principales relations auxquelles on peut s'intéresser dans un modèle statistique :
- `y ~ x` : régression simple,
- `y ~ x + 0` : idem avec suppression du terme d'ordonnée à l'origine,
- `y ~ a + b` : régresse avec deux effets principaux indépendants,
- `y ~ a * b` : idem avec interaction (équivalent à `1 + a + b + a:b`),
- `y ~ a / b` : idem en considérant une relation d'emboîtement (équivalent à `1 + a + b + a %in% b`).
L'opérateur `|` est quant à lui utilisé par l'extension `lme4`{.pkg} dans le cadre de modèles mixtes avec effets aléatoires.
Voir le chapitre dédié à la [régression logistique](regression-logistique.html) pour des exemples de modèles multivariés et le chapitre dédié aux [effets d'interaction](effets-d-interaction.html) pour plus de détails sur cette notion.
## Pour aller plus loin
Comme vient de le voir dans ce chapitre, la notation <q>formule</q> apparaît çà et là dans les différentes fonctions de **R** est de ses extensions. Il est par conséquent utile d'en connaître les rudiments, et en particulier les opérateurs `~` (_tilde_) et `+`, ne serait-ce que pour pouvoir se servir des différentes fonctions présentées sur cette page. Le chapitre [lattice et les formules](lattice-graphiques-et-formules.html) fournit plus de détails sur ces aspects.
La notation <q>formule</q> devient cruciale dès que l'on souhaite rédiger des modèles : la formule `y ~ x`, par exemple, qui est équivalente à la formule `y ~ 1 + x`, correspond à l'équation mathématique $Y = a + bX$. On trouvera de nombreux exemples d'usage de cette notation dans les chapitres consacrés, notamment, à la régression linéaire ou à la [régression logistique](regression-logistique.html).
De la même manière, l'opérateur `|` (_pipe_) utilisé par l'extension `lattice`{.pkg} joue aussi un rôle très important dans la rédaction de modèles multi-niveaux, où il sert à indiquer les variables à pentes ou à coefficients aléatoires. Ces modèles sont présentés dans un [chapitre dédié](modeles-lineaires-a-effets-mixtes.html).