I would like to make my code more efficient, I have a survey where my data looks like:
survey <- data.frame(
x = c(1, 6, 2, 60, 75, 40, 27, 10),
y = c(100, 340, 670, 700, 450, 200, 136, 145))
#Two lists:
A <- c(3, 6, 7, 27, 40, 41)
t <- c(0.10, 0.11, 0.12, 0.13, 0.14, 0.15, 0.16)
What I did was create new columns, like this:
z <- ifelse(survey$x %in% A), 0, min(t))
for (i in t) {
survey[paste0("T",i)] <-z
survey[paste0("T",i)] <-ifelse (z > 0, i, z)
}
But with that code it takes a while, is there a better way to do it?
As the OP mentioned about speed of execution, the data.table
option would be faster
library(data.table)
i1 <- !survey$x %in% A
setDT(survey)[, paste0("T", t) := 0]
for(j in t) {
set(survey2, i = which(i1), j = paste0("T", j), value = j)
}
set.seed(24)
survey1 <- data.frame(x = sample(survey$x, 1e7, replace = TRUE),
y = sample(survey$y, 1e7, replace = TRUE))
survey2 <- copy(survey1)
system.time({
survey1[paste0("T", t)] <- lapply(t, function(y) ifelse(survey1$x %in% A, 0, y))
})
# user system elapsed
# 8.20 2.75 11.03
system.time({
i1 <- !survey2$x %in% A
setDT(survey2)[, paste0("T", t) := 0]
for(j in t) {
set(survey2, i = which(i1), j = paste0("T", j), value = j)
}
})
# user system elapsed
# 0.97 0.31 1.28
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With