Reputation: 33
Suppose I have two similar big objects x, y (datatable) defined in an envirnomnet e. I would like to change a big part of x or y in a similar way using a function f without creating a copy of x or y within the executive environment of f. Example:
e <- new.env()
e$x <- c(1,2,3) # imagine this to be BIG (ie. dataframe with 200k vars each 500k rows)
e$y <- c(4,5,6) # same here
e$v <- 2 # minor variables
f <- function(var_str, env, input){
# do some computation on parts of var_str which is either "x" or "y"
# and store these right back into e$x or e$y, respectively.
# ie
str <- paste0(var_str,"[2:3] <- (",var_str,"[2:3])^2 + rep(v,2) + ", deparse(input1),"^3/c(100,101)")
eval(parse(text=str), envir= e)
# this does work but I can image there is an easier/more elegant way
# of doing this.
}
I would like to define the function in the global environment and apply this function to e$x and e$y with different variables in input. Ie. execute
f("x", e, c(1,2))
f("y", e, c(3,4))
Does anybody have an elegant solution to this.
Upvotes: 2
Views: 1315
Reputation: 263332
Out of curiosity I tried working with the simple example in the R6 tutorial and came up with this (copied as a console transcript). I honestly didn't know if it complies with the non-copy requirements of the request, but it does appear to modify an object in-place.
# Assumes one has created an R6 constructor named `RC` ...
# slightly extended from the example in section
RC <- setRefClass("RC",
fields = list(x = 'ANY'),
methods = list(
getx = function() x,
setx = function(value) x <<- value,
setsub = function(i,j,val) x[i,j] <<- val
)
)
#--- execution ---
rc <- RC$new()
rc$setx(matrix(1:20, 4,5))
rc
# --- result ---
Reference class object of class "RC"
Field "x":
[,1] [,2] [,3] [,4] [,5]
[1,] 1 5 9 13 17
[2,] 2 6 10 14 18
[3,] 3 7 11 15 19
[4,] 4 8 12 16 20
Test the setsub
class-specific function:
rc$setsub(4,5,0) #Test of setting a single element within an object to a new value
rc
#-------
Reference class object of class "RC"
Field "x":
[,1] [,2] [,3] [,4] [,5]
[1,] 1 5 9 13 17
[2,] 2 6 10 14 18
[3,] 3 7 11 15 19
[4,] 4 8 12 16 0
rc$setsub(1:4,1:4,0) #Test of setting a range of elements within an object to a new value
rc
# ---------
Reference class object of class "RC"
Field "x":
[,1] [,2] [,3] [,4] [,5]
[1,] 0 0 0 0 17
[2,] 0 0 0 0 18
[3,] 0 0 0 0 19
[4,] 0 0 0 0 0
So this (very lightly tested) implementation of the [<-
-function via setsub
does succeed. A single effort to extend this to the example using a self
reference did fail.
Upvotes: 0
Reputation: 132706
eval(parse())
is to be avoided. You can reference the environment like this:
e <- new.env()
e$x <- c(1,2,3) # imagine this to be BIG (ie. dataframe with 200k vars each 500k rows)
e$y <- c(4,5,6) # same here
e$v <- 2 # minor variables
f <- function(var_str, env, input){
# do some computation on parts of var_str which is either "x" or "y"
# and store these right back into e$x or e$y, respectively.
# ie
env[[var_str]][2:3] <- (env[[var_str]][2:3])^2 + rep(env$v,2) + input^3/input
}
f("x", e, 1:2)
e$x
#[1] 1 7 15
Upvotes: 3