R apply family
参考资料:A brief introduction to “apply” in R
部分更新内容来自 R Cookbook。
1. lapply
: Apply a Function to Each Element of a List or Vector
lapply(X, func, ...)
可以理解成:
List<T> result = ...;
for (T xn : X) {
result.add(func(xn, ...));
}
return result;
If X is not a list, it will be coerced to a list using as.list
.
if (!is.vector(X) || is.object(X))
X <- as.list(X)
lapply always returns a list, regardless of the class of the input.
apply family 里常见 anonymous function,比如这个 lapply(x, function(x) x[,1])
就是取 list<Matrix>
中每个 matrix 的第一列。
2. sapply
: Simplify the Result of lapply
The simplification rule is:
- If the function returns a list where every element is length 1, then a vector is returned
- If the function returns a list where every element is a vector of the same length (> 1), a matrix is returned.
- If it can’t figure things out, a list is returned
举个高级一点的例子,假设 scores
是一个 list,包含 4 个 vector 分别是某课程 4 个 semester 的成绩,要求对每个 vector 做 t-test:
> tests <- lapply(scores, t.test) ## 如果用 sapply,返回 matrix 就不好办了
> sapply(tests, function(t) t$conf.int) ## function 的作用就是把 t$conf.int 给 print 出来
还有个有点巧妙的用法:查看 data frame 每个 column 的 class:
> sapply(batches, class)
batch clinic dosage shrinkage
"factor" "factor" "integer" "numeric"
2.1 sapply
example: Removing low-correlation variables from a set of predictors
Suppose that resp
is a response variable (a vector) and pred
is a data frame of predictor variables. Suppose further that we have too many predictors and therefore want to select the top 10 as measured by correlation with the response.
The first step is to calculate the correlation between each predictor and response. In R, that’s a one-liner:
> cors <- sapply(pred, cor, y=resp)
Any arguments beyond the second one in sapply
are passed to cor
, so the function call will be cor(pred[[i]],y=resp)
, which calculates the correlation between the given column and resp
.
The result cors
is a vector of correlations, one for each column. We use the rank
function to find the positions of the correlations that have the largest magnitude:
> mask <- (rank(-abs(cors)) <= 10)
rank
的作用是把 vector 的元素按升序排列,返回一个序号 vector,比如
> rank(c(4,6,5))
[1] 1 3 2 ## 表示 4 是一号位,6 是三号位,5 是二号位
我们知道 abs(cors)
是越大越相关,于是 -abs(cors)
是越小越相关,给 -abs(cors)
排序的话排在前头的都是小值,所以 rank(-abs(cors)) <= 10
是前 10 位,也就是最小的 10 个 -abs(cors)
的位置,也就是最相关的 10 个 predictor 的位置(有点绕,自己体会下)。
Using mask
, we can select just those columns from the data frame:
> best.pred <- pred[,mask]
At this point, we can regress resp
against best.pred
, knowing that we have chosen the predictors with the highest correlations:
> lm(resp ~ best.pred)
2.2 vapply
: Safer sapply
vapply
is similar to sapply
, but has a pre-specified type of return value, so it can be safer (and sometimes faster) to use.
3. mapply
: Apply a Function to Parallel Vectors or Lists (a Multivariate Version of sapply
)
举个例子:
> l1 <- list(a = c(1:10), b = c(11:20))
> l2 <- list(c = c(21:30), d = c(31:40))
> mapply(sum, l1$a, l1$b, l2$c, l2$d)
[1] 64 68 72 76 80 84 88 92 96 100
注意,这里 mapply
并不是:
sapply(l1$a, sum)
sapply(l1$b, sum)
sapply(l2$c, sum)
sapply(l2$d, sum)
而是:
for (int i = 1; i <= 10; ++i) {
list.add(sum(l1$a[i], l1$b[i], l2$c[i], l2$d[i]));
}
return list;
注意 mapply
的 function 要求是 works on scalars but not on vectors。
mapply
可以用于多个 vector 也可以用于多个 list:
> mapply(f, vec1, vec2, ..., vecN)
> mapply(f, list1, list2, ..., listN)
4. apply
: Apply a Function over Array Margins (e.g. to Every Row or to Every Column)
apply(X, MARGIN, FUN, ...)
首先我们要搞清楚 R 的 array。在 R 中说 array 你不能直接联想到 int[]
,因为 R 的 array 上来就是多维的,而且你最好理解为多维 matrix。单个的 matrix 可以看做是最简单的 array。下面这个 array 你可以理解成 4 个 matrix,想象成 4 页纸,每张纸上有一个 matrix;或者想象成 4 块玻璃板,每一块上有一个 matrix,4 块玻璃板拼成一个 matrix 立方体。
> x <- array(rep(1, 24), c(2, 3, 4))
> x
, , 1
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 2
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 3
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 4
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
然后再是这个 “Array Margins”,这个名字起得很奇怪,从字面上很难理解,我们举两个例子说明下:
> x <- matrix(rep(1, 6), nrow=2, ncol=3)
> x
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
> apply(x, 1, sum)
[1] 3 3
> apply(x, 2, sum)
[1] 2 2 2
> apply(x, c(1, 2), sum)
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
对 matrix 而言,margin = 1
就是 apply by row,margin = 2
就是 apply by column,此时 the function being called should expect one argument, a
vector, which will be one row or one column from the matrix;如果 margin = c(1, 2)
就是 apply by every single element,此时 function 就只需要接收 single element 作为参数。
对 data frame 而言,如果你要 apply by column,其实可以不用 apply(margin=2)
这么麻烦(and in this case R will convert your data frame to a
matrix and then apply your function),直接用 lapply
或者 sapply
就行,因为 data frame 本质上是一个 list,list 的元素就是它的 column。
> x <- array(rep(1, 24), c(2, 3, 4))
> x
, , 1
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 2
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 3
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
, , 4
[,1] [,2] [,3]
[1,] 1 1 1
[2,] 1 1 1
> apply(x, 1, sum)
[1] 12 12
> apply(x, 2, sum)
[1] 8 8 8
> apply(x, 3, sum)
[1] 6 6 6 6
> apply(x, c(1, 2), sum)
[,1] [,2] [,3]
[1,] 4 4 4
[2,] 4 4 4
> apply(x, c(1, 3), sum)
[,1] [,2] [,3] [,4]
[1,] 3 3 3 3
[2,] 3 3 3 3
> apply(x, c(2, 3), sum)
[,1] [,2] [,3] [,4]
[1,] 2 2 2 2
[2,] 2 2 2 2
[3,] 2 2 2 2
立体的情况复杂一点,请发挥你的空间想象能力~
For sums and means of matrix dimensions, we have some shortcuts.
rowSums
=apply(x, 1, sum)
rowMeans
=apply(x, 1, mean)
colSums
=apply(x, 2, sum)
colMeans
=apply(x, 2, mean)
The shortcut functions are much faster,因为有专门优化过.
5. tapply
: Apply a Function over a Ragged Array (i.e. lapply after splitting a column)
function (X, INDEX, FUN = NULL, ..., simplify = TRUE)
- X: an atomic object, typically a vector.
- INDEX: list of one or more factors, each of same length as X. The elements are coerced to factors by
as.factor
. - FUN: the function to be applied, or NULL. In the case of functions like +, %*%, etc., the function name must be backquoted or quoted. If FUN is NULL, tapply returns a vector which can be used to subscript the multi-way array tapply normally produces.
- simplify: If FALSE, tapply always returns an array of mode “list”. If TRUE (the default), then if FUN always returns a scalar, tapply returns an array with the mode of the scalar.
又有新概念了…… “Ragged Array”。其实这个在 Java 里也有,也叫 “Jagged Array”,就是指子数组不整齐的多维数组,比如 { {1, 2}, {3, 4, 5} }
这样的。R 里的 Ragged Array 也是这个意思,所以这里的 Array 又不是多维 matrix 的那个 Array(你们多想个名字出来会死啊!)
然后我们的 tapply
并不是直接作用在 Ragged Array 上的,这个 Ragged Array 是由 X 和 INDEX 两个参数拼起来的。以最简单的情况,X 是 vector、INDEX 是 factor 举个例子:
> X <- 1:9
> INDEX <- factor('a', 'a', 'a', 'a', 'b', 'b', 'b', 'c', 'c')
这两个参数一拼就会形成:
- a: 1, 2, 3, 4
- b: 5, 6, 7
- c: 8, 9
这就是所谓的 Ragged Array。manual 也有说:
The combination of a vector and a labelling factor is an example of what is sometimes called a ragged array, since the subclass sizes are possibly irregular. When the subclass sizes are all the same the indexing may be done implicitly and much more efficiently.
一个超级好的类比是 Histogram:
- a: ========
- b: ======
- c: ==
然后我们算下按 a、b、c 分类的 sum:
> tapply(X, INDEX, sum)
a b c
10 18 17
说白了就是 tapply(X, INDEX, fun)
== lapply(split(X, INDEX), fun)
,我们先用 split
来对某一个 column 做 grouping,得到一个 list of vectors,也就是 list of groups,然后对这个 list of groups 做 lapply
6. split
: Split a Vector (or list) or Data Frame into Groups by a Factor or List of Factors
最常见的就是 data frame 中有一个 column 是 factor,我们称其为 grouping factor。split(x,y)
的意思就是 split x by factor y into a list of vectors
。
从另一个角度来说,split
就是 tapply
拼 Ragged Array 的过程,举个例子:
> X <- 1:30
> INDEX <- gl(3, 10) ## Generate Levels:10 个 1,10 个 2,10 个 3;levels = 1, 2, 3
> split(X, INDEX)
$`1`
[1] 1 2 3 4 5 6 7 8 9 10
$`2`
[1] 11 12 13 14 15 16 17 18 19 20
$`3`
[1] 21 22 23 24 25 26 27 28 29 30
tapply(X, INDEX, fun)
== lapply(split(X, INDEX), fun)
> lapply(split(X, INDEX), sum)
$`1`
[1] 55
$`2`
[1] 155
$`3`
[1] 255
下面看一个按两个 factor 分组的例子:
> X <- 1:10
> INDEX_1 <- as.factor(c(rep('a', 5), rep('b', 5)))
> INDEX_2 <- gl(5, 2)
> INDEX_1
[1] a a a a a b b b b b
Levels: a b
> INDEX_2
[1] 1 1 2 2 3 3 4 4 5 5
Levels: 1 2 3 4 5
> str(split(X, INDEX_1))
List of 2
$ a: int [1:5] 1 2 3 4 5
$ b: int [1:5] 6 7 8 9 10
> str(split(X, INDEX_2))
List of 5
$ 1: int [1:2] 1 2
$ 2: int [1:2] 3 4
$ 3: int [1:2] 5 6
$ 4: int [1:2] 7 8
$ 5: int [1:2] 9 10
> str(split(X, list(INDEX_1, INDEX_2)))
List of 10
$ a.1: int [1:2] 1 2
$ b.1: int(0)
$ a.2: int [1:2] 3 4
$ b.2: int(0)
$ a.3: int 5
$ b.3: int 6
$ a.4: int(0)
$ b.4: int [1:2] 7 8
$ a.5: int(0)
可见 X$m.n
== X$m
∩ X$n
。
drop = TRUE
的作用是去掉空行:
> str(split(X, list(INDEX_1, INDEX_2), drop=TRUE))
List of 6
$ a.1: int [1:2] 1 2
$ a.2: int [1:2] 3 4
$ a.3: int 5
$ b.3: int 6
$ b.4: int [1:2] 7 8
$ b.5: int [1:2] 9 10
Alternatively, you can use the unstack
function:
> groups <- split(x, f)
> groups <- unstack(data.frame(x,f))
Both functions return a list of vectors, where each vector contains the elements for one group.
The unstack
function goes one step further: if all vectors have the same length, it converts the list into a data frame.
7. by
: Apply a Function to Groups of Rows (i.e. lapply
after splitting a data frame)
split
一个 column 得到一个 list of vectors,split
一个 data frame 会得到一个 list of data frames。所以 by(dfrm, factor, fun)
就是先 split
这个 dfrm
by factor
,然后在得到的 list of data frames 上 lapply
执行 fun
。与 tapply
很像,我们可以直接理解为:by(dfrm, factor, fun)
== lapply(split(dfrm, factor), fun)
。
这里 function 就必须是接收 data frame 为参数,一个常见的符合条件的 function 就是 summary
,这也是常见的组合用法,比如:
> by(trials, trials$sex, summary)
高级一点的例子是 “分组 Linear Regression”:
> models <- by(trials, trials$sex, function(df) lm(post~pre+dose1+dose2, data=df)) ## `models` is a list of linear models
> lapply(models, confint) ## print confidence intervals of each linear model
Comments