mixgb: Multiple Imputation Through XGBoost
Yongshi Deng
2024-11-30
Source:vignettes/Using-mixgb.Rmd
Using-mixgb.Rmd
Introduction
Mixgb offers a scalable solution for imputing large datasets using XGBoost, subsampling and predictive mean matching. Our method utilizes the capabilities of XGBoost, a highly efficient implementation of gradient boosted trees, to capture interactions and non-linear relations automatically. Moreover, we have integrated subsampling and predictive mean matching to minimize bias and reflect appropriate imputation variability. Our package supports various types of variables and offers flexible settings for subsampling and predictive mean matching. We also include diagnostic tools for evaluating the quality of the imputed values.
Impute missing values with mixgb
We first load the mixgb
package and the
nhanes3_newborn
dataset, which contains 16 variables of
various types (integer/numeric/factor/ordinal factor). There are 9
variables with missing values.
library(mixgb)
str(nhanes3_newborn)
#> tibble [2,107 × 16] (S3: tbl_df/tbl/data.frame)
#> $ HSHSIZER: int [1:2107] 4 3 5 4 4 3 5 3 3 3 ...
#> $ HSAGEIR : int [1:2107] 2 5 10 10 8 3 10 7 2 7 ...
#> $ HSSEX : Factor w/ 2 levels "1","2": 2 1 2 2 1 1 2 2 2 1 ...
#> $ DMARACER: Factor w/ 3 levels "1","2","3": 1 1 2 1 1 1 2 1 2 2 ...
#> $ DMAETHNR: Factor w/ 3 levels "1","2","3": 3 1 3 3 3 3 3 3 3 3 ...
#> $ DMARETHN: Factor w/ 4 levels "1","2","3","4": 1 3 2 1 1 1 2 1 2 2 ...
#> $ BMPHEAD : num [1:2107] 39.3 45.4 43.9 45.8 44.9 42.2 45.8 NA 40.2 44.5 ...
#> ..- attr(*, "label")= chr "Head circumference (cm)"
#> $ BMPRECUM: num [1:2107] 59.5 69.2 69.8 73.8 69 61.7 74.8 NA 64.5 70.2 ...
#> ..- attr(*, "label")= chr "Recumbent length (cm)"
#> $ BMPSB1 : num [1:2107] 8.2 13 6 8 8.2 9.4 5.2 NA 7 5.9 ...
#> ..- attr(*, "label")= chr "First subscapular skinfold (mm)"
#> $ BMPSB2 : num [1:2107] 8 13 5.6 10 7.8 8.4 5.2 NA 7 5.4 ...
#> ..- attr(*, "label")= chr "Second subscapular skinfold (mm)"
#> $ BMPTR1 : num [1:2107] 9 15.6 7 16.4 9.8 9.6 5.8 NA 11 6.8 ...
#> ..- attr(*, "label")= chr "First triceps skinfold (mm)"
#> $ BMPTR2 : num [1:2107] 9.4 14 8.2 12 8.8 8.2 6.6 NA 10.9 7.6 ...
#> ..- attr(*, "label")= chr "Second triceps skinfold (mm)"
#> $ BMPWT : num [1:2107] 6.35 9.45 7.15 10.7 9.35 7.15 8.35 NA 7.35 8.65 ...
#> ..- attr(*, "label")= chr "Weight (kg)"
#> $ DMPPIR : num [1:2107] 3.186 1.269 0.416 2.063 1.464 ...
#> ..- attr(*, "label")= chr "Poverty income ratio"
#> $ HFF1 : Factor w/ 2 levels "1","2": 2 2 1 1 1 2 2 1 2 1 ...
#> $ HYD1 : Ord.factor w/ 5 levels "1"<"2"<"3"<"4"<..: 1 3 1 1 1 1 1 1 2 1 ...
colSums(is.na(nhanes3_newborn))
#> HSHSIZER HSAGEIR HSSEX DMARACER DMAETHNR DMARETHN BMPHEAD BMPRECUM
#> 0 0 0 0 0 0 124 114
#> BMPSB1 BMPSB2 BMPTR1 BMPTR2 BMPWT DMPPIR HFF1 HYD1
#> 161 169 124 167 117 192 7 0
To impute this dataset, we can use the default settings. The default
number of imputed datasets is m = 5
. Note that we do not
need to convert our data into dgCMatrix or one-hot coding format. Our
package will automatically convert it for you. Variables should be of
the following types: numeric, integer, factor or ordinal factor.
# use mixgb with default settings
imputed.data <- mixgb(data = nhanes3_newborn, m = 5)
Customize imputation settings
We can also customize imputation settings:
The number of imputed datasets
m
The number of imputation iterations
maxit
XGBoost hyperparameters and verbose settings.
xgb.params
,nrounds
,early_stopping_rounds
,print_every_n
andverbose
.Subsampling ratio. By default,
subsample = 0.7
. Users can change this value under thexgb.params
argument.Predictive mean matching settings
pmm.type
,pmm.k
andpmm.link
.Whether ordinal factors should be converted to integer (imputation process may be faster)
ordinalAsInteger
Whether or not to use bootstrapping
bootstrap
Initial imputation methods for different types of variables
initial.num
,initial.int
andinitial.fac
.Whether to save models for imputing newdata
save.models
andsave.vars
.
# Use mixgb with chosen settings
params <- list(
max_depth = 5,
subsample = 0.9,
nthread = 2,
tree_method = "hist"
)
imputed.data <- mixgb(
data = nhanes3_newborn, m = 10, maxit = 2,
ordinalAsInteger = FALSE, bootstrap = FALSE,
pmm.type = "auto", pmm.k = 5, pmm.link = "prob",
initial.num = "normal", initial.int = "mode", initial.fac = "mode",
save.models = FALSE, save.vars = NULL,
xgb.params = params, nrounds = 200, early_stopping_rounds = 10, print_every_n = 10L, verbose = 0
)
Tune hyperparameters
Imputation performance can be affected by the hyperparameter
settings. Although tuning a large set of hyperparameters may appear
intimidating, it is often possible to narrowing down the search space
because many hyperparameters are correlated. In our package, the
function mixgb_cv()
can be used to tune the number of
boosting rounds - nrounds
. There is no default
nrounds
value in XGBoost,
so users are
required to specify this value themselves. The default
nrounds
in mixgb()
is 100. However, we
recommend using mixgb_cv()
to find the optimal
nrounds
first.
params <- list(max_depth = 3, subsample = 0.7, nthread = 2)
cv.results <- mixgb_cv(data = nhanes3_newborn, nrounds = 100, xgb.params = params, verbose = FALSE)
cv.results$evaluation.log
#> iter train_rmse_mean train_rmse_std test_rmse_mean test_rmse_std
#> <num> <num> <num> <num> <num>
#> 1: 1 5.6243200 0.01725215 5.6242119 0.07519387
#> 2: 2 3.9873035 0.01041157 3.9927615 0.07643722
#> 3: 3 2.8486556 0.01272129 2.8524742 0.07121946
#> 4: 4 2.0598481 0.01185795 2.0647573 0.07232445
#> 5: 5 1.5196993 0.01574637 1.5293888 0.07642496
#> 6: 6 1.1555977 0.01745843 1.1770621 0.08059807
#> 7: 7 0.9218320 0.02051193 0.9529445 0.09058948
#> 8: 8 0.7719014 0.02131583 0.8155824 0.09712346
#> 9: 9 0.6840419 0.02121464 0.7364341 0.10134552
#> 10: 10 0.6319835 0.02324912 0.6910356 0.10442807
#> 11: 11 0.6013581 0.02490061 0.6666511 0.10520464
#> 12: 12 0.5835209 0.02668848 0.6527954 0.10396770
#> 13: 13 0.5702095 0.02693038 0.6454456 0.10355826
#> 14: 14 0.5594641 0.02447102 0.6417038 0.10258203
#> 15: 15 0.5540570 0.02305729 0.6379671 0.10187290
#> 16: 16 0.5504931 0.02227745 0.6371141 0.10272587
#> 17: 17 0.5421563 0.02074779 0.6349845 0.10188686
#> 18: 18 0.5395515 0.02107327 0.6333410 0.10147670
#> 19: 19 0.5342802 0.01954349 0.6361272 0.09976709
#> 20: 20 0.5309925 0.01900489 0.6362062 0.10066105
#> 21: 21 0.5277205 0.01834704 0.6367092 0.09997261
#> 22: 22 0.5255634 0.01808838 0.6369874 0.09927022
#> 23: 23 0.5218592 0.01784562 0.6380108 0.09842193
#> 24: 24 0.5177798 0.01593962 0.6380967 0.09938728
#> 25: 25 0.5133888 0.01539523 0.6387354 0.09861629
#> 26: 26 0.5103064 0.01560169 0.6386598 0.09888282
#> 27: 27 0.5081871 0.01583676 0.6400786 0.09894321
#> 28: 28 0.5058566 0.01618987 0.6401878 0.09903052
#> iter train_rmse_mean train_rmse_std test_rmse_mean test_rmse_std
cv.results$response
#> [1] "BMPWT"
cv.results$best.nrounds
#> [1] 18
By default, mixgb_cv()
will randomly choose an
incomplete variable as the response and build an XGBoost model with
other variables as explanatory variables using the complete cases of the
dataset. Therefore, each run of mixgb_cv()
will likely
return different results. Users can also specify the response and
covariates in the argument response
and
select_features
respectively.
cv.results <- mixgb_cv(
data = nhanes3_newborn, nfold = 10, nrounds = 100, early_stopping_rounds = 1,
response = "BMPHEAD", select_features = c("HSAGEIR", "HSSEX", "DMARETHN", "BMPRECUM", "BMPSB1", "BMPSB2", "BMPTR1", "BMPTR2", "BMPWT"), xgb.params = params, verbose = FALSE
)
cv.results$best.nrounds
#> [1] 20
Let us just try setting
nrounds = cv.results$best.nrounds
in mixgb()
to obtain 5 imputed datasets.
imputed.data <- mixgb(data = nhanes3_newborn, m = 5, nrounds = cv.results$best.nrounds)
Inspect multiply imputed values
The mixgb
package used to provide a few visual
diagnostics functions. However, we have moved these functions to the
vismi
package, which provides a wide range of visualisation
tools for multiple imputation.
For more details, please check the vismi
package on
GitHub Visualisation Tools
for Multiple Imputation.