-
Notifications
You must be signed in to change notification settings - Fork 51
/
block04_dataAggregation.rmd
355 lines (280 loc) · 20.5 KB
/
block04_dataAggregation.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
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
Data aggregation
========================================================
### Optional getting started advice
*Ignore if you don't need this bit of support.*
This is one in a series of tutorials in which we explore basic data import, exploration and much more using data from the [Gapminder project](http://www.gapminder.org). Now is the time to make sure you are working in the appropriate directory on your computer, perhaps through the use of an [RStudio project](block01_basicsWorkspaceWorkingDirProject.html). To ensure a clean slate, you may wish to clean out your workspace and restart R (both available from the RStudio Session menu, among other methods). Confirm that the new R process has the desired working directory, for example, with the `getwd()` command or by glancing at the top of RStudio's Console pane.
Open a new R script (in RStudio, File > New > R Script). Develop and run your code from there (recommended) or periodicially copy "good" commands from the history. In due course, save this script with a name ending in .r or .R, containing no spaces or other funny stuff, and evoking "data aggregation".
### Load the Gapminder data
Assuming the data can be found in the current working directory, this works:
```{r, eval=FALSE}
gDat <- read.delim("gapminderDataFiveYear.txt")
```
Plan B (I use here, because of where the source of this tutorial lives):
```{r}
## data import from URL
gdURL <- "http://www.stat.ubc.ca/~jenny/notOcto/STAT545A/examples/gapminder/data/gapminderDataFiveYear.txt"
gDat <- read.delim(file = gdURL)
```
Basic sanity check that the import has gone well:
```{r}
str(gDat)
```
### Data aggregation
If you feel the urge to store a little snippet of a data.frame:
```{r}
(snippet <- subset(gDat, country == "Canada"))
```
Stop and ask yourself ...
> Do I want to create sub-data.frames for each level of some factor (or unique combination of several factors) ... in order to compute or graph something?
If NO, then maybe you really do need to store a copy of a subset of the data.frame. But seriously consider whether you can achieve your goals by simply using the `subset =` argument -- or perhaps `with()` coupled with `subset()` -- to enact a computation on a specific set of rows. If this still does not suit your needs, then maybe you really should use `subset()` as shown above and carry on.
If YES, __use data aggregation techniques__ or conditioning in `lattice` or facetting in `ggplot2` plots -- _don’t subset the data.frame_. Or, to be totally clear, only subset the data.frame as a temporary measure as you develop your elegant code for computing on or visualizing these sub-data.frames.
#### Data aggregation landscape
There are two main options for data aggregation:
* built-in functions, often referred to as the `apply` family of functions
* the [`plyr`](http://plyr.had.co.nz) add-on package
I used the built-in functions for many years but am transitioning to `plyr`. I recommend simply starting with `plyr` if you are new to R. You can see older material about data aggregation with built-in functions [here](block93_oldDataAggregation.html). In this tutorial we will only use `plyr`.
You'll notice I did not even mention another option that may occur to some: hand-coding `for` loops, perhaps, even (shudder) nested `for` loops! Don't do it. By the end of this tutorial you'll see things that are much faster and more fun. Yes, of course, tedious loops are required for data aggregation but when you can, let other developers write them for you, in super-efficient low level code. This is more about saving programmer time than compute time, BTW.
#### Install and load `plyr`
If you have not already done so, you'll need to install `plyr`. Here's one way to do that:
```{r, eval = FALSE}
install.packages("plyr", dependencies = TRUE)
```
You will also need to load the package before you can use the functions in an R session:
```{r}
library(plyr)
```
You can [make that apply to all your R sessions via your `.Rprofile`](http://stackoverflow.com/questions/10300769/how-to-load-packages-in-r-automatically). Note this is a controversial practice, because it means your code will not necessarily run "as is" on someone else's computer. For that reason, I auto-load packages *very sparingly*. `plyr` is not (yet) on my list.
#### `plyr` Big Ideas
The `plyr` functions will not make much sense viewed individually, e.g. simply reading the help for `ddply()` is not the fast track to competence. There is a very important over-arching logic for the package and it is well worth reading the article [The split-apply-combine strategy for data analysis](http://www.jstatsoft.org/v40/i01/paper), Hadley Wickham, Journal of Statistical Software, vol. 40, no. 1, pp. 1–29, 2011. Though it is no substitute for reading the above, here is the most critical information:
* __split-apply-combine__: A common analytical pattern is to split data into logical bits, apply some function to each bit, and stick the results back together again. Recognize when you're solving such a problem and exploit the right tools.
* The computations on these little bits must be truly independent, i.e. the problem must be [embarrassingly or pleasingly parallel](http://en.wikipedia.org/wiki/Embarrassingly_parallel), in order to use `plyr`.
* The heart of `plyr` is a set a functions with names like this: `XYply` where `X` specifies what sort of input you're giving and `Y` specifies the sort of output you want.
- `a` = array, where matrices and vectors are important special cases
- `d` = data.frame
- `l` = list
- `_` = no output; only valid for `Y`, obviously; useful when you're operating on a list purely for the side effects, e.g., making a plot or sending output to screen/file
* The usage is very similar across these functions. Here are the main arguments:
- `.data` is the first argument = the input
- the next argument specifies how to split up the input into bits; it is does not exist when the input is a list, because the pieces are obviously the list components
- then comes the function and further arguments needed to describe the computation to be applied to the bits
Today we will emphasize `ddply()` which accepts a data.frame, splits it into pieces based on one or more factors, computes on the pieces, then returns the results as a data.frame. For the record, the built-in functions most relevant to `ddply()` are `tapply()` and friends.
#### `ddply()`
Let's say we want to get the maximum life expectancy for each continent.
```{r, tidy=FALSE}
(maxLeByCont <- ddply(gDat, ~ continent, summarize, maxLifeExp = max(lifeExp)))
```
Let's study the return value.
```{r}
str(maxLeByCont)
levels(maxLeByCont$continent)
```
So we got a data.frame back, with one observation per continent, and two variables: the maximum life expectancies and the continent, as a factor, with the same levels in the same order, as for the input data.frame `gDat`. If you have sweated to do such things with built-in functions, this minor miracle might make you cry tears of joy (or anguish over all the hours you have wasted.)
`summarize()` or its synonym `summarise()` is a function provided by `plyr` that creates a new data.frame from an old one. It is related to the built-in function `transform()` that transforms variables in a data.frame or adds new ones. Feel free to play with it a bit in some top-level commands; you will use it alot inside `plyr` calls.
The two variables in `maxLeByCont` come from two sources. The `continent` factor is provided by `ddply()` and represents the labelling of the life expectancies with their associated continent. This is the book-keeping associated with dividing the input into little bits, computing on them, and gluing the results together again in an orderly, labelled fashion. We can take more credit for the other variable `maxLifeExp`, which has a name we chose ("maxLifeExp") and arises from applying a function we specified (`max()`) to a variable of our choice (`lifeExp`).
**You try:** compute the minimum GDP per capita by continent. Here's what I get:
```{r, eval=TRUE, echo=FALSE}
ddply(gDat, ~ continent, summarize, minGdpPercap = min(gdpPercap))
```
You might have chosen a different name for the minimum GDP/capita's, but your numerical results should match.
The function you want to apply to the continent-specific data.frames can be built-in, like `max()` above, or a custom function you've written. This custom function can be written in advance or specified 'on the fly'. Here's how I would count the number of countries in this dataset for each continent.
```{r}
ddply(gDat, ~ continent, summarize, nUniqCountries = length(unique(country)))
```
Here is another way to do the same thing that doesn't use `summarize()` at all:
```{r, tidy=FALSE}
ddply(gDat, ~ continent,
function(x) return(c(nUniqCountries = length(unique(x$country)))))
```
In pseudo pseudo-code, here is what's happening in both of the above commands:
```{r, eval=FALSE, results='asis', tidy=FALSE}
returnValue <- an empty receptacle with one "slot" per country
for each possible country i {
x <- subset(gDat, subset = country == i)
returnValue[i] <- length(unique(x$country))
name or label for returnValue[i] is set to country i
}
ddply packages returnValue and associate names/labels as a nice data.frame
```
You don't have to compute just one thing for each sub-data.frame, nor are you limited to computing on just one variable. Check it out.
```{r, tidy=FALSE}
ddply(gDat, ~ continent, summarize,
minLifeExp = min(lifeExp), maxLifeExp = max(lifeExp),
medGdpPercap = median(gdpPercap))
```
### Putting it all together: using `ddply()` and polishing the results
Now I want to do something more complicated. I want to fit a linear regression for each country, modelling life expectancy as a function of the year and then retain the estimated intercepts and slopes. I will walk before I run. Therefore, I will create a tiny sub-data.frame to prototype this, before I fold it into a `ddply()` call. If you're a newbie, watch how complicated tasks are slowly constructed.
```{r}
jCountry <- "France" # pick, but do not hard wire, an example
(jDat <- subset(gDat, country == jCountry)) # temporary measure!
xyplot(lifeExp ~ year, jDat, type = c("p", "r")) # always plot the data
jFit <- lm(lifeExp ~ year, jDat)
summary(jFit)
```
Wow, check out that crazy intercept! Apparently the life expectancy in France around year 0 A.D. was minus 400 years! This a great opportunity for some sanity checking of a model fit and thinking about how to reparametrize the model to make the parameters have natural interpretation. I think it makes more sense for the intercept to correspond to life expectancy in 1952, the earliest date in our dataset. Let's try that again.
```{r}
(yearMin <- min(gDat$year))
jFit <- lm(lifeExp ~ I(year - yearMin), jDat)
summary(jFit)
```
An intercept around 68 years makes much more common sense and is also supported by our plot. What is this `jFit` object and how can I get stuff out of it?
```{r}
class(jFit)
mode(jFit)
```
It turns out `jFit` is of class "lm" and its mode is list. So that means I could use indexing to isolate specific components. But what's in there?
```{r}
## str(jFit) # too ugly to print here but you should look
names(jFit)
jFit$coefficients
```
Using `str()` and `names()` reveals a great deal about this "lm" object and reading the help file for `lm()` would explain a great deal more. In the See Also section we learn there's a generic function `coef()` which looks promising.
```{r}
coef(jFit)
```
As a rule, I use extractor functions like this when they are available.
```{r, eval=FALSE, echo=FALSE}
methods(class = "lm")
```
We have achieved our goal for this specific country -- we've gotten its intercept and slope. Now we need to package that as a function (we will talk about functions properly later, but this should be fairly self-explanatory).
```{r}
jFun <- function(x) coef(lm(lifeExp ~ I(year - yearMin), x))
jFun(jDat) # trying out our new function ... yes still get same numbers
```
I hate the names of these return values. Good names pay off downstream, so I will enhance my function.
```{r}
jFun <- function(x) {
estCoefs <- coef(lm(lifeExp ~ I(year - yearMin), x))
names(estCoefs) <- c("intercept", "slope")
return(estCoefs)
}
jFun(jDat) # trying out our improved function ... yes still get same numbers
```
It's always a good idea to try out a function on a few small examples.
```{r}
jFun(subset(gDat, country == "Canada"))
jFun(subset(gDat, country == "Uruguay"))
jFun(subset(gDat, country == "India"))
```
It seems like we are ready to scale up by placing this function inside a `ddply()` call.
```{r}
jCoefs <- ddply(gDat, ~ country, jFun)
str(jCoefs)
tail(jCoefs)
```
We did it! By the time we've packaged the computation in a function, the call itself is deceptively simple. To review, here's the script I would save from our work in this section:
```{r}
## realistically, you would read the data from a local file
gdURL <- "http://www.stat.ubc.ca/~jenny/notOcto/STAT545A/examples/gapminder/data/gapminderDataFiveYear.txt"
gDat <- read.delim(file = gdURL)
## str(gDat) here when working interactively
yearMin <- min(gDat$year)
jFun <- function(x) {
estCoefs <- coef(lm(lifeExp ~ I(year - yearMin), x))
names(estCoefs) <- c("intercept", "slope")
return(estCoefs)
}
## jFun(subset(gDat, country == "India")) to see what it does
jCoefs <- ddply(gDat, ~ country, jFun)
```
Over the course of its development, the number of lines in the script would start small, get bigger as I fiddle around, make mistakes, write and use lots of sanity checking code and then .... contract down to the above, as I strip it down to bare necessities. That is how the pros actually work. They don't write beautiful elegant scripts the first time and they aren't satisfied with hideous hack-y "transcripts" of the very first attempt that (sort of) worked.
Finally, let's present this information attractively in a table. *I consider my advice about how to do this less definitive than the above. I'll be happy to see people explore other table-making tools.* Fixed width plain text printing of data.frames is OK for internal use and during development. But at some point you will want a nicer looking table. Markdown doesn't have a proper table syntax, because it is a ruthlessly simple language. Some Markdown dialects / processing engines support table extensions, BTW. But here I will teach you how to make an HTML table. This HTML code will survive unharmed as your R Markdown document is converted from R Markdown to Markdown and finally to HTML.
I've experimented with the `xtable` package, which you will need to install
```{r, eval = FALSE}
install.packages("xtable", dependencies = TRUE)
```
and load.
```{r}
library(xtable)
```
Let's pick some countries at random and display their estimated coefficients. *FYI: the following R chunk has an option `results='asis'` and that is important to the correct display of the table.*
```{r results='asis'}
set.seed(916)
foo <- jCoefs[sample(nrow(jCoefs), size = 15), ]
foo <- xtable(foo)
print(foo, type='html', include.rownames = FALSE)
```
Two easy improvments to make this table more useful are
* include the continent information
* sort it rationally
The easiest way to get the continent information is to enhance our `ddply()` call. Here is what we used:
```{r, eval=FALSE}
jCoefs <- ddply(gDat, ~ country, jFun)
```
This divides `gDat` into country-specific pieces. But we can supply two factors in the second argument: `country` and `continent`. In theory, sub-data.frames will be made for all possible combinations of the levels of `country` and `continent`. Many of those will have zero rows because there is, for example, no Belgium in Asia. By default, `plyr` functions drop these empty combinations. But the labelling work done by `ddply()` will still help us, as we will get both `country` and `continent` as factors in our result. This is easier to see than explain!
```{r}
jCoefs <- ddply(gDat, ~ country + continent, jFun)
str(jCoefs)
tail(jCoefs)
```
Now, prior to making the HTML table, we will sort the data.frame, so it starts with the country with the shortest life expectancy in 1952, and goes to the largest.
```{r results='asis'}
set.seed(916)
foo <- jCoefs[sample(nrow(jCoefs), size = 15), ]
foo <- arrange(foo, intercept)
## foo <- foo[order(foo$intercept), ] # an uglier non-plyr way
foo <- xtable(foo)
print(foo, type='html', include.rownames = FALSE)
```
Soon we will start making the companion plots ... but for now our work is done.
### Lessons
`plyr` is a powerful package for data aggregation.
`ddply()` is the most important function: data.frames are great, therefore `ddply()` is great because it takes data.frame as input and returns data.frame as output.
Simple functions, built-in or user-defined, can be provided directly in a call to `ddply()`.
It's better to handle more complicated data aggregation differently. Build it up slowly and revisit earlier steps for improvement. Here is a gentle workflow:
* Create an indicative sub-data.frame
* Compute on it directly with top-level commands until you achieve your goal
* Package those commands inside a function
* Test the function with other sub-data.frames
* Refine your function, e.g. give the return values better names
* Construct a `ddply()` call using your function
* Identify more weakness in your function, refine, call `ddply()` again
Results you present to the world generally need polishing to make the best impression and have the best impact. This too will be an iterative process. Here are some good things to think about:
* Is there an obvious piece of ancillary information that I should include? Example: continent above. It was not needed for the computation but it is helpful to retain/restore it for the final table.
* Is there some logical ordering for the data? Alphabetical is ususally the default and it is completely arbitrary and often confusing.
* Have I presented the data in the most aesthetically pleasing way I currently know how?
### Exporting data
What it you wanted to write the table of slopes and intercepts to file? Go to the tutorial on [getting data out of R](block05_getNumbersOut.html), but if this is an emergency, the main functions to consider are `write.table()` and `saveRDS()`.
### References
`plyr` paper: [The split-apply-combine strategy for data analysis](http://www.jstatsoft.org/v40/i01/paper), Hadley Wickham, Journal of Statistical Software, vol. 40, no. 1, pp. 1–29, 2011. Go [here](http://www.jstatsoft.org/v40/i01/) for supplements, such as example code from the paper.
[Data Manipulation with R](http://www.springerlink.com/content/t19776/?p=0ecea4f02a68458eb3d605ec3cdfc7ef%CF%80=0) by Phil Spector, Springer (2008) | [author webpage](http://www.stat.berkeley.edu/%7Espector/) | [GoogleBooks search](http://books.google.com/books?id=grfuq1twFe4C&lpg=PP1&dq=data%2520manipulation%2520spector&pg=PP1#v=onepage&q=&f=false)
* The main link above to SpringerLink will give full access to the book if you are on a UBC network (or any other network that confers accesss).
* See Chapter 8 (“Data Aggregation”)
### Q & A
Student: How do you pass more than one argument for a function into `ddply()`. The main example that we used in class was this:
```{r}
(yearMin <- min(gDat$year))
jFun <- function(x) {
estCoefs <- coef(lm(lifeExp ~ I(year - yearMin), x))
names(estCoefs) <- c("intercept", "slope")
return(estCoefs)
}
jCoefs <- ddply(gDat, ~country, jFun)
head(jCoefs)
```
and `jFun` only requires one argument, `x`. What if it had more than one argument?
Answer: Let's imagine that the shift for the year covariate is an argument instead of a previously-assigned variable `yearMin`. Here's how it would work.
```{r}
jFunTwoArgs <- function(x, cvShift = 0) {
estCoefs <- coef(lm(lifeExp ~ I(year - cvShift), x))
names(estCoefs) <- c("intercept", "slope")
return(estCoefs)
}
```
Since I've assigned `cvShift =` a default value of zero, we can get coefficients where the intercept corresponds to the year A.D. 0 with this simple call:
```{r}
jCoefsSilly <- ddply(gDat, ~ country, jFunTwoArgs)
head(jCoefsSilly)
```
We are getting the same estimated slopes but the silly year 0 intercepts we've seen before. Let's use the `cvShift =` argument to resolve this.
```{r}
jCoefsSane <- ddply(gDat, ~ country, jFunTwoArgs, cvShift = 1952)
head(jCoefsSane)
```
We're back to our usual estimated intercepts, which reflect life expectancy in 1952. Of course hard-wiring 1952 is not a great idea, so here's probably our best code yet:
```{r}
jCoefsBest <- ddply(gDat, ~ country, jFunTwoArgs, cvShift = min(gDat$year))
head(jCoefsBest)
```
<div class="footer">
This work is licensed under the <a href="http://creativecommons.org/licenses/by-nc/3.0/">CC BY-NC 3.0 Creative Commons License</a>.
</div>