Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

can the value.var in dcast be a list or have multiple value variables?

In the help files for dcast.data.table, there is a note stating that a new feature has been implemented: "dcast.data.table allows value.var column to be of type list"

I take this to mean that one can have multiple value variables within a list, i.e. in this format:

dcast.data.table(dt, x1~x2, value.var=list('var1','var2','var3')) 

But we get an error: 'value.var' must be a character vector of length 1.

Is there such a feature, and if not, what would be other one-liner alternatives?

EDIT: In reply to the comments below

There are situations where you have multiple variables that you want to treat as the value.var. Imagine for example that x2 consists of 3 different weeks, and you have 2 value variables such as salt and sugar consumption and you want to cast those variables across the different weeks. Sure, you can 'melt' the 2 value variables into a single column, but why do something using two functions, when you can do it in one function like reshape does?

(Note: I've also noticed that reshape cannot treat multiple variables as the time variable as dcast does.)

So my point is that I don't understand why these functions don't allow for the flexibility to include multiple variables within the value.var or the time.var just as we allow for multiple variables for the id.var.

like image 340
AlexR Avatar asked Apr 14 '14 09:04

AlexR


People also ask

What is value var in dcast?

dcast uses formula interface. The variables on the LHS of formula represents the id vars and RHS the measure vars. value. var denotes the column to be filled in with while casting to wide format.

What does dcast do in R?

dcast: Convert data between wide and long forms.


2 Answers

From v1.9.6 of data.table, we can cast multiple value.var columns simultaneously (and also use multiple aggregation functions in fun.aggregate). Please see ?dcast and the Efficient reshaping using data.tables vignette for more.

Here's how we could use dcast:

dcast(setDT(mydf), x1 ~ x2, value.var=c("salt", "sugar")) #    x1 salt_1 salt_2 salt_3 sugar_1 sugar_2 sugar_3 # 1:  1      3      4      6       1       2       2 # 2:  2     10      3      9       5       3       6 # 3:  3     10      7      7       4       6       7 
like image 166
Arun Avatar answered Oct 02 '22 10:10

Arun


Update

Apparently, the fix was much easier...


Technically, your statement that "apparently there is no such feature" isn't quite correct. There is such a feature in the recast function (which sort of hides the melting and casting process), but it seems like Hadley forgot to finish the function or something: the function returns a list of the relevant parts of your operation.

Here's a minimal example...

Some sample data:

set.seed(1) mydf <- data.frame(x1 = rep(1:3, each = 3),                    x2 = rep(1:3, 3),                    salt = sample(10, 9, TRUE),                    sugar = sample(7, 9, TRUE))  mydf #   x1 x2 salt sugar # 1  1  1    3     1 # 2  1  2    4     2 # 3  1  3    6     2 # 4  2  1   10     5 # 5  2  2    3     3 # 6  2  3    9     6 # 7  3  1   10     4 # 8  3  2    7     6 # 9  3  3    7     7 

The effect you seem to be trying to achieve:

reshape(mydf, idvar='x1', timevar='x2', direction='wide') #   x1 salt.1 sugar.1 salt.2 sugar.2 salt.3 sugar.3 # 1  1      3       1      4       2      6       2 # 4  2     10       5      3       3      9       6 # 7  3     10       4      7       6      7       7 

recast in action. (Note that the values are all what we would expect in the dimensions we would expect it.)

library(reshape2) out <- recast(mydf, x1 ~ x2 + variable, measure.var = c("salt", "sugar")) ### recast(mydf, x1 ~ x2 + variable, id.var = c("x1", "x2")) out # $data #      [,1] [,2] [,3] [,4] [,5] [,6] # [1,]    3    1    4    2    6    2 # [2,]   10    5    3    3    9    6 # [3,]   10    4    7    6    7    7 #  # $labels # $labels[[1]] #   x1 # 1  1 # 2  2 # 3  3 #  # $labels[[2]] #   x2 variable # 1  1     salt # 2  1    sugar # 3  2     salt # 4  2    sugar # 5  3     salt # 6  3    sugar 

I'm honestly not sure if this was an incomplete function, or if it is a helper function to another function.

All of the information is there to be able to put the data back together again, making it easy to write a function like this:

recast2 <- function(...) {   inList <- recast(...)   setNames(cbind(inList[[2]][[1]], inList[[1]]),            c(names(inList[[2]][[1]]),               do.call(paste, c(rev(inList[[2]][[2]]), sep = "_")))) } recast2(mydf, x1 ~ x2 + variable, measure.var = c("salt", "sugar")) #   x1 salt_1 sugar_1 salt_2 sugar_2 salt_3 sugar_3 # 1  1      3       1      4       2      6       2 # 2  2     10       5      3       3      9       6 # 3  3     10       4      7       6      7       7 

Again, a possible advantage with the recast2 approach is the ability to aggregate as well as reshape in the same step.

like image 32
A5C1D2H2I1M1N2O1R2T1 Avatar answered Oct 02 '22 09:10

A5C1D2H2I1M1N2O1R2T1