Here is a better way to go around this:
rv <- rev(v)
f <- function(vec, pos){z<-rep(NA,length(vec)+1);z[pos]<-0;z[is.na(z)]<-vec;z;}
t(sapply(5:1, function(x) {if (x%%2==1) f(rv,x) else f(v,x)}))
# [,1] [,2] [,3] [,4] [,5]
# [1,] 8 6 4 2 0
# [2,] 2 4 6 0 8
# [3,] 8 6 0 4 2
# [4,] 2 0 4 6 8
# [5,] 0 8 6 4 2
Function f
, inserts 0
into vec
at the specified position pos
.
BENCHMARKING
library(microbenchmark)
v <- c(2, 4, 6, 8)
rv <- rev(v)
f <- function(vec, pos){z<-rep(NA,length(vec)+1);z[pos]<-0;z[is.na(z)]<-vec;z;}
f_m0h3n <- function(v){t(sapply(5:1, function(x) {if (x%%2==1) f(rv,x) else f(v,x)}))}
f_Roland <- function(v){
m <- matrix(0, length(v), length(v) + 1)
m[, c(FALSE, TRUE)] <- rev(v)
m[, c(TRUE, FALSE)] <- v
m <- t(m)
m1 <- matrix(0, length(v) + 1, length(v) + 1)
m1[upper.tri(m1)] <- m[upper.tri(m, TRUE)]
m1[lower.tri(m1)] <- m[lower.tri(m)]
m1[, rev(seq_len(ncol(m1)))]
}
all(f_m0h3n(v)==f_Roland(v))
# [1] TRUE
microbenchmark(f_m0h3n(v), f_Roland(v))
# Unit: microseconds
# expr min lq mean median uq max neval
# f_m0h3n(v) 106.931 109.4975 115.1818 112.064 119.7625 180.927 100
# f_Roland(v) 114.202 116.3410 128.5183 119.763 126.8210 430.290 100