在今天的答案中引用的@hadley 关于函数的文章的提示下,我决定重新审视一个关于outer
函数如何工作(或不工作)的持久性难题。为什么会失败:
outer(0:5, 0:6, sum) # while outer(0:5, 0:6, "+") succeeds
这显示了我认为outer
应该如何处理如下函数sum
:
Outer <- function(x,y,fun) {
mat <- matrix(NA, length(x), length(y))
for (i in seq_along(x)) {
for (j in seq_along(y)) {mat[i,j] <- fun(x[i],y[j])} }
mat}
> Outer(0:5, 0:6, `+`)
[,1] [,2] [,3] [,4] [,5] [,6] [,7]
[1,] 0 1 2 3 4 5 6
[2,] 1 2 3 4 5 6 7
[3,] 2 3 4 5 6 7 8
[4,] 3 4 5 6 7 8 9
[5,] 4 5 6 7 8 9 10
[6,] 5 6 7 8 9 10 11
好的,我的索引并没有与该示例完全对齐,但修复起来并不难。问题是为什么像这样的函数sum
应该能够接受两个参数并返回适合矩阵元素的(原子)值,而在传递给base::outer
函数时却不能这样做?
所以@agstudy 为更紧凑的版本提供了灵感,Outer
他的版本更加紧凑:
Outer <- function(x,y,fun) {
mat <- matrix(mapply(fun, rep(x, length(y)),
rep(y, each=length(x))),
length(x), length(y))
然而,问题仍然存在。术语“矢量化”在这里有些模棱两可,我认为“二元”更正确,因为sin
并且cos
在该术语的通常意义上是“矢量化”的。outer
期望以可以使用非二元函数的方式扩展其论点是否存在基本的逻辑障碍。
这是另一个outer
错误,可能与我对这个问题缺乏了解类似:
> Vectorize(sum)
function (..., na.rm = FALSE) .Primitive("sum")
> outer(0:5, 0:6, function(x,y) Vectorize(sum)(x,y) )
Error in outer(0:5, 0:6, function(x, y) Vectorize(sum)(x, y)) :
dims [product 42] do not match the length of object [1]