Search code examples
rexperimental-design

Randomizing a split-plot (and other classical designs) in R


My question is whether any R packages out there provide function(s) that make it reasonably easy to randomize a standard experimental design that may involve crossed factors, nesting, and/or blocking.

For concreteness, show me specifically how to produce a new randomization of the Oats experiment provided as a dataset in the nlme package.

> data(Oats, package="nlme")
> summary(Oats)
 Block           Variety       nitro          yield      
 VI :12   Golden Rain:24   Min.   :0.00   Min.   : 53.0  
 V  :12   Marvellous :24   1st Qu.:0.15   1st Qu.: 86.0  
 III:12   Victory    :24   Median :0.30   Median :102.5  
 IV :12                    Mean   :0.30   Mean   :104.0  
 II :12                    3rd Qu.:0.45   3rd Qu.:121.2  
 I  :12                    Max.   :0.60   Max.   :174.0 

In that experiment, there are six blocks. Each block is divided into three plots which are randomly assigned to varieties (one plot per variety in each block, each block separately randomized). Each plot is subdivided into 4 subplots, and randomly assigned to four amounts of nitrogen (0, 0.2, 0.4, and 0.6), one subplot per nitro level randomized separately in each plot. In the dataset, the plots are identifiable as the combinations of Block and Variety. (The response variable is yield, so that is not really part of the treatment design.)

Second question: Given you can randomize Oats, can you use the same package to easily randomize other classical designs, e.g., a three-factor CRD, a nested design, a three-period crossover design, or a 5x5 Graeco-Latin square?

I actually already know how to do this using base functions in the R language; so I'm not particularly interested in seeing a programmatic answer. I want to know if any existing packages make this easy. I can identify a few packages that may help, e.g., randomizeR and randomizr, but a quick reading of the documentation of these still does not make it very obvious (to me) how to do this.

I have the makings of a general-purpose randomization package I developed a few years ago for my students, and I am trying to decide whether to develop it further for release on CRAN.


Solution

  • Here's how I'd do it with randomizr:

    data(Oats, package="nlme")
    
    # get the latest version from github      
    install.packages("devtools")
    devtools::install_github("DeclareDesign/randomizr")
    
    library(randomizr)
    Oats <- within(Oats,{
      Variety_new <- block_ra(block_var = Block, 
                              condition_names = c("Golden Rain", "Marvellous", "Victory"))
      nitro_new <- block_ra(block_var = paste0(Block, Variety_new), 
                            condition_names = c(0, 0.2, 0.4, 0.6))
      })
    
    # Original Random Assignment
    with(Oats, table(Block, Variety))
    with(Oats, table(Block, nitro))
    with(Oats, table(Block, nitro, Variety))
    
    # New Random Assignment
    with(Oats, table(Block, Variety_new))
    with(Oats, table(Block, nitro_new))
    with(Oats, table(Block, nitro_new, Variety_new))
    

    The key is to realize that you need to block on both variety and Block to randomize the subplots into nitro conditions. (that's why we need the call to paste0).

    EDIT

    here is another way to do this (see comments)

    library(randomizr) 
    des <- rev(expand.grid(subplot=1:4, wholeplot=1:3, block=1:6)) 
    des <- within(des,{ 
       plot_id <- paste0(block, "_", wholeplot) 
       Variety <- block_and_cluster_ra(
                    block_var = block, 
                    clust_var = plot_id, 
                    condition_names = c("Golden Rain", "Marvellous", "Victory")) 
       nitro <- block_ra(block_var = plot_id, 
                         condition_names = c(0, 0.2, 0.4, 0.6)) 
    }) 
    
    with(des, table(Variety, block)) 
    with(des, table(Variety, nitro)) 
    with(des, table(Variety, nitro, block))