knitr::opts_chunk$set( collapse = TRUE, comment = "#>" )
In the context of educational large-scale assessments (but also in other contexts) we frequently encounter data sets which have an hierarchical structure. In educational large-scale assessments these can, for example, be pupils nested in schools. Additional, hidden nested structures occur, if missing data are treated with multiple imputations or person parameters are estimated using plausible values. In these cases it is inefficient to store all the data in one rectangular data set. In other data science applications the use of relational data bases is a widespread measure to tackle this challenge.
eatGADS
supports creating such relational data bases (based on the open source software SQLite3
and the R
package eatDB
) while maintaining its meta data and providing a very user friendly interface for data users who are unfamiliar with relational data bases. In doing so, it allows the handling of large data sets even on limited hardware settings. Furthermore, this approach allows the extraction of data from different hierarchy levels, which means that data has to be reshaped very rarely.
This vignette illustrates how a relational eatGADS
data base can be created from a rectangular SPSS
(.sav
) data file. For illustrative purposes we use a small example data set from the campus files of the German PISA Plus assessment. The complete campus files and the original data set can be accessed here and here.
library(eatGADS)
We can import an .sav
(or an compressed .zsav
) data set via the import_spss()
function. Checks on variable names for SQLite3
compliance are performed automatically. Changes to the variable names are reported to the console.
sav_path <- system.file("extdata", "pisa.zsav", package = "eatGADS") dat <- import_spss(sav_path)
The next steps depend on the data structure: If the different hierarchy levels are saved in different source data sets (e.g. different .sav
files) the next section can be skipped. However, sometimes data from different hierarchy levels is saved in one data file. Then, splitting and reshaping becomes necessary.
In this case, we want to split the imported GADSdat
object into its hierarchy levels (in our example: background data on level 1 and imputed competence data on level 2). This can be achieved by the splitGADS()
function. We specify the hierarchical structure as a list
. After this, we can extract separate GADSdat
objects by name via the extractGADS()
function. These objects can then be used for reshaping.
For reasons of simplicity, the example only contains two hierarchy levels. In practice, often more hierarchy levels are present. Splitting can be performed into as many hierarchy levels as desired. The reshaping has to be performed for each hierarchy level separately.
pvs <- grep("pv", namesGADS(dat), value = T) splitted_gads <- splitGADS(dat, nameList = list(noImp = namesGADS(dat)[!namesGADS(dat) %in% pvs], PVs = c("idstud", pvs))) # new Structure namesGADS(splitted_gads) # Extract GADSdat objects noImp_gads <- extractGADSdat(splitted_gads, "noImp") pvs_gads <- extractGADSdat(splitted_gads, "PVs")
For reshaping data we highly recommend the R
package tidyr
. Its performance might be less optimized than for example the data.table
package, however it is very intuitive and user friendly. For our example data set we need to reshape the PVs
from wide to long format and then separate the resulting column into two columns, containing the dimension
and imputation number (imp
) (Note: This results in a data set in which different dimensions for a single student are stored in separate rows, not columns). For this, we directly access the data in the GADSdat
object via pvs_gads$dat
. The reshaping is performed by tidyr::pivot_longer()
. tidyr::separate()
is used to separate our two additional identifier columns (dimension
and imp
). Finally, we clean the imp
column and transform it to numeric.
# Extract raw data from pv gads pvs_wide <- pvs_gads$dat # Wide format head(pvs_wide) pvs_long1 <- tidyr::pivot_longer(pvs_wide, cols = all_of(pvs)) pvs_long2 <- tidyr::separate(pvs_long1, col = "name", sep = "_", into = c("dimension", "imp")) pvs_long2$imp <- as.numeric(gsub("pv", "", pvs_long2$imp)) # Finale long format head(as.data.frame(pvs_long2))
After reshaping we adapt the meta data in our initial GADSdat
object via updateMeta()
. This is necessary, as variables have been removed from the data set (e.g. "ma_pv1"
etc.) and new variables have replaced them ("value"
, "dimension"
, "imp"
). Now we have to add some variable labels, as most of the old variable labels got lost due to the reshaping. For an extensive tutorial see the vignette Handling Meta Data.
pvs_gads2 <- updateMeta(pvs_gads, newDat = as.data.frame(pvs_long2)) extractMeta(pvs_gads2) # pvs_gads2 <- changeVarLabels(pvs_gads2, varName = c("dimension", "imp", "value"), varLabel = c("Achievement dimension (math, reading, science)", "Number of imputation of plausible values", "Plausible Value")) extractMeta(pvs_gads2)
For the creation of a relational data base we recreate the initial hierarchical structure via mergeLabels()
(which performs the reverse action as extractGADS()
). Furthermore, we create two lists, a primary key list (pkList
) and a foreign key list (fkList
). Primary keys are the variables that uniquely identify each row within every hierarchy level. Foreign keys are the variables that allow merging between different hierarchy levels. In the list of foreign keys we also have to specify another hierarchy level, to which each hierarchy level connects. An exception is the lowest hierarchy levels, which serves as a basis.
By setting up the order and the foreign keys of the data base we specify how the data is merged together when we extract data from it. In contrast to conventional relational data bases, eatGADS
data bases are less flexible: The package does not support modifying the data base after its creation or extracting data from it with different merges than specified when it was set up.
merged_gads <- mergeLabels(noImp = noImp_gads, PVs = pvs_gads2) pkList <- list(noImp = "idstud", PVs = c("idstud", "imp", "dimension")) fkList <- list(noImp = list(References = NULL, Keys = NULL), PVs = list(References = "noImp", Keys = "idstud"))
Finally, we create the relational data base on disc via the createGADS()
function.
temp_path <- paste0(tempfile(), ".db") createGADS(merged_gads, pkList = pkList, fkList = fkList, filePath = temp_path)
For a detailed tutorial on how to use a relational eatGADS
data base see the vignette getGADS
: Using a relational eatGADS data base.
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.