personalized
To demonstrate how to use the personalized
package with
multi-category treatments, we simulate data with three treatments. The
treatment assignments will be based on covariates and hence mimic an
observational setting with no unmeasured confounders.
library(personalized)
set.seed(123)
<- 250
n.obs <- 10
n.vars <- matrix(rnorm(n.obs * n.vars, sd = 3), n.obs, n.vars)
x
# simulated non-randomized treatment with multiple levels
# based off of a multinomial logistic model
<- 0.1 + 0.5 * x[,1] - 0.25 * x[,5]
xbetat_1 <- 0.1 - 0.5 * x[,9] + 0.25 * x[,5]
xbetat_2 1.prob <- exp(xbetat_1) / (1 + exp(xbetat_1) + exp(xbetat_2))
trt.2.prob <- exp(xbetat_2) / (1 + exp(xbetat_1) + exp(xbetat_2))
trt.3.prob <- 1 - (trt.1.prob + trt.2.prob)
trt.
<- cbind(trt.1.prob, trt.2.prob, trt.3.prob)
prob.mat <- apply(prob.mat, 1, function(rr) rmultinom(1, 1, prob = rr))
trt.mat <- apply(trt.mat, 2, function(rr) which(rr == 1))
trt.num <- as.factor(paste0("Trt_", trt.num))
trt
# simulate response
# effect of treatment 1 relative to treatment 3
<- 2 * (0.5 + x[,2] - 2 * x[,3] )
delta1 # effect of treatment 2 relative to treatment 3
<- (0.5 + x[,6] - 2 * x[,5] )
delta2
# main covariate effects with nonlinearities
<- x[,1] + x[,9] - 2 * x[,4]^2 + x[,4] +
xbeta 0.5 * x[,5] ^ 2 + 2 * x[,2] - 3 * x[,5]
# create entire functional form of E(Y|T,X)
<- xbeta +
xbeta * ((trt.num == 1) - (trt.num == 3) ) +
delta1 * ((trt.num == 2) - (trt.num == 3) )
delta2
# simulate continuous outcomes E(Y|T,X)
<- xbeta + rnorm(n.obs, sd = 2) y
We will use the version of the treatment status vector in our analysis, however, the integer values vector, i.e. , could be used as well.
1:5] trt[
## [1] Trt_3 Trt_3 Trt_1 Trt_1 Trt_1
## Levels: Trt_1 Trt_2 Trt_3
table(trt)
## trt
## Trt_1 Trt_2 Trt_3
## 97 86 67
Then we construct a propensity score function that takes covariate information and the treatment statuses as input and generate a matrix of probabilities as output. Each row \(i\) of the output matrix represents an observation and each column \(j\) is the probability that the \(i\)th patient received the \(j\)th treatment. The treatment levels are ordered alphabetically (or numerically if the treatment assignment vector is a vector of integers). Our propensity score model in this example will be a multinomial logistic regression model with a lasso penalty for the probability of treatment assignments conditional on covariate information:
<- function(x, trt)
propensity.multinom.lasso
{if (!is.factor(trt)) trt <- as.factor(trt)
<- cv.glmnet(y = trt, x = x, family = "multinomial",
gfit nfolds = 3)
# predict returns a matrix of probabilities:
# one column for each treatment level
<- drop(predict(gfit, newx = x,
propens type = "response", s = "lambda.min"))
# return the matrix probability of treatment assignments
<- propens[,match(levels(trt), colnames(propens))]
probs
probs }
An important assumption for the propensity score is that \(0 < Pr(T_i = t | \mathbf X) < 1\) for all \(\mathbf X\) and \(t\). This assumption, often called the positivity assumption, is impossible to verify. However, in practice validity of the assumption can be assessed via a visualization of the empirical overlap of our estimated propensity scores to determine if there is any evidence of positivity violations. The function also allows us to visualize the overlap of our propensity scores for multi-category treatment applications. The following code results in the plot shown Figure \(\ref{fig:check_overlap_multitreat}\).
check.overlap(x = x, trt = trt, propensity.multinom.lasso)
Each plot above is for a different treatment group, e.g. the plot in the first row of plots is the subset of patients who received treatment 1. There seems to be no obvious evidence against the positivity assumption.
As the outcome is continuous and there is a large number of covariates available for our construction of a benefit score, we will use the squared error loss and a lasso penalty. The model can be fit in the same manner as for the binary treatment setting, however only linear models and the weighting method are available. Here we can also specify the reference treatment (the treatment that the non-reference treatments are compared with by each benefit score).
set.seed(123)
<- fit.subgroup(x = x, y = y,
subgrp.multi trt = trt, propensity.func = propensity.multinom.lasso,
reference.trt = "Trt_3",
loss = "sq_loss_lasso",
nfolds = 3)
summary(subgrp.multi)
## family: gaussian
## loss: sq_loss_lasso
## method: weighting
## cutpoint: 0
## propensity
## function: propensity.func
##
## benefit score: f_Trt_1(x): Trt_1 vs Trt_3, f_Trt_2(x): Trt_2 vs Trt_3
## f_Trt_3(x): 0
## maxval = max(f_Trt_1(x), f_Trt_2(x))
## which.max(maxval) = The trt level which maximizes maxval
## Trt recom = which.max(maxval)*I(maxval > c) + Trt_3*I(maxval <= c) where c is 'cutpoint'
##
## Average Outcomes:
## Recommended Trt_1 Recommended Trt_2 Recommended Trt_3
## Received Trt_1 -8.2279 (n = 97) NaN (n = 0) NaN (n = 0)
## Received Trt_2 -22.7982 (n = 86) NaN (n = 0) NaN (n = 0)
## Received Trt_3 -15.5278 (n = 67) NaN (n = 0) NaN (n = 0)
##
## Treatment effects conditional on subgroups:
## Est of E[Y|T=Trt_1,Recom=Trt_1]-E[Y|T=/=Trt_1,Recom=Trt_1]
## 10.8514 (n = 250)
## Est of E[Y|T=Trt_2,Recom=Trt_2]-E[Y|T=/=Trt_2,Recom=Trt_2]
## NaN (n = 0)
## Est of E[Y|T=Trt_3,Recom=Trt_3]-E[Y|T=/=Trt_3,Recom=Trt_3]
## NaN (n = 0)
##
## NOTE: The above average outcomes are biased estimates of
## the expected outcomes conditional on subgroups.
## Use 'validate.subgroup()' to obtain unbiased estimates.
##
## ---------------------------------------------------
##
## Benefit score 1 quantiles (f(X) for Trt_1 vs Trt_3):
## 0% 25% 50% 75% 100%
## 7.17 7.17 7.17 7.17 7.17
##
## Benefit score 2 quantiles (f(X) for Trt_2 vs Trt_3):
## 0% 25% 50% 75% 100%
## -13.2904 -8.2866 -6.5613 -5.2289 0.0758
##
## ---------------------------------------------------
##
## Summary of individual treatment effects:
## E[Y|T=trt, X] - E[Y|T=Trt_3, X]
## where 'trt' is Trt_1 and Trt_2
##
## Trt_1-vs-Trt_3 Trt_2-vs-Trt_3
## Min. :14.34 Min. :-26.5807
## 1st Qu.:14.34 1st Qu.:-16.5731
## Median :14.34 Median :-13.1227
## Mean :14.34 Mean :-13.3502
## 3rd Qu.:14.34 3rd Qu.:-10.4579
## Max. :14.34 Max. : 0.1516
##
## ---------------------------------------------------
##
## 2 out of 20 interactions selected in total by the lasso (cross validation criterion).
##
## The first estimate is the treatment main effect, which is always selected.
## Any other variables selected represent treatment-covariate interactions.
##
##
## 0 out of 10 variables selected for delta 1 by the lasso (cross validation criterion).
##
## Trt_1
## Estimates for delta (Trt_1 vs Trt_3) 7.1698
##
## 2 out of 10 variables selected for delta 2 by the lasso (cross validation criterion).
##
## Trt_2 V5 V9
## Estimates for delta (Trt_2 vs Trt_3) -6.9472 -0.0497 -0.9131
The function now displays selected variables for each of the two benefit scores and shows the quantiles of each benefit score. We can also plot the empirical observations within the different subgroups using the function, however now it is slightly more complicated. It appears that the average outcome is higher for those who received the level of the treatment they were recommended than those who received a different treatment than they were recommended. Also note that returns a object and can thus be modified by the user. The below example yields Figure \(\ref{fig:plot_multi_trt_model}\).
<- plot(subgrp.multi)
pl + theme(axis.text.x = element_text(angle = 90, hjust = 1)) pl
To obtain valid estimates of the subgroup-specific treatment effects,
we perform the repeated training and testing resample procedure using
the function (here we set the number of replications B=4
,
but this should be a much larger number in practice) :
set.seed(123)
<- validate.subgroup(subgrp.multi,
validation.multi B = 4, # specify the number of replications
method = "training_test_replication",
train.fraction = 0.5)
print(validation.multi, digits = 2, sample.pct = TRUE)
## family: gaussian
## loss: sq_loss_lasso
## method: weighting
##
## validation method: training_test_replication
## cutpoint: 0
## replications: 4
##
## benefit score: f_Trt_1(x): Trt_1 vs Trt_3, f_Trt_2(x): Trt_2 vs Trt_3
## f_Trt_3(x): 0
## maxval = max(f_Trt_1(x), f_Trt_2(x))
## which.max(maxval) = The trt level which maximizes maxval
## Trt recom = which.max(maxval)*I(maxval > c) + Trt_3*I(maxval <= c) where c is 'cutpoint'
##
## Average Test Set Outcomes:
## Recommended Trt_1 Recommended Trt_2
## Received Trt_1 -2.86 (SE = 1.35, 22.6%) -2.54 (SE = 9.3, 13.4%)
## Received Trt_2 -25.56 (SE = 13.08, 24%) -6.8 (SE = 32.93, 9.2%)
## Received Trt_3 -15.41 (SE = 2.1, 16%) -25.13 (SE = 7.82, 7.8%)
## Recommended Trt_3
## Received Trt_1 -36.2 (SE = 10.77, 3.2%)
## Received Trt_2 -43.89 (SE = 25.4, 2.4%)
## Received Trt_3 -21.62 (SE = 9.74, 1.4%)
##
## Treatment effects conditional on subgroups:
## Est of E[Y|T=Trt_1,Recom=Trt_1]-E[Y|T=/=Trt_1,Recom=Trt_1]
## 18.38 (SE = 6.9, 62.6%)
## Est of E[Y|T=Trt_2,Recom=Trt_2]-E[Y|T=/=Trt_2,Recom=Trt_2]
## 5.1 (SE = 29.95, 30.4%)
## Est of E[Y|T=Trt_3,Recom=Trt_3]-E[Y|T=/=Trt_3,Recom=Trt_3]
## 19.33 (SE = 9.61, 7%)
##
## Est of
## E[Y|Trt received = Trt recom] - E[Y|Trt received =/= Trt recom]:
## 8.3 (SE = 16.44)
Setting the argument above to prints out the average percent of all patients which are in each subgroup (as opposed to the average sample sizes). We can see that about 58% of patients were recommended treatment 1 and among those recommended treatment 1, we expect them to have larger outcomes if they actually receive treatment 1 as opposed to the other treatments. The estimated effects are positive within all three subgroups (meaning those recommended each of the different treatments have a positive benefit from receiving the treatment they are recommended as opposed to receiving any another treatments).
We can visualize the subgroup-specific treatment effects using as usual with results shown in Figure \(\ref{fig:plotcomparemultivalidated}\):
<- plot(validation.multi)
plv + theme(axis.text.x = element_text(angle = 90, hjust = 1)) plv
For cases with multi-category treatments, the user must specify a propensity function that returns \(Pr(T = T_i | \mathbf X= \mathbf x)\) for patient \(i\). In other words, it should return the probability of receiving the treatment that was actually received for each patient. For example:
<- function(x, trt)
propensity.func.multinom
{<- data.frame(trt = trt, x)
df <- nnet::multinom(trt ~ . -trt, data = df)
mfit # predict returns a matrix of probabilities:
# one column for each treatment level
<- nnet::predict.nnet(mfit, type = "probs")
propens
if (is.factor(trt))
{<- levels(trt)[trt]
values else
}
{<- trt
values
}# return the probability corresponding to the
# treatment that was observed
<- propens[cbind(1:nrow(propens),
probs match(values, colnames(propens)))]
probs }
Optionally the user can specify the function to return a matrix of treatment probabilities, however, the columns must be ordered by the levels of . An example of this is the following:
<- function(x, trt)
propensity.func.multinom
{require(nnet)
<- data.frame(trt = trt, x)
df <- multinom(trt ~ . -trt, data = df)
mfit # predict returns a matrix of probabilities:
# one column for each treatment level
<- predict(mfit, type = "probs")
propens
if (is.factor(trt))
{<- levels(trt)
levels else
}
{<- sort(unique(trt))
levels
}# return the probability corresponding to the
# treatment that was observed
<- propens[,match(levels, colnames(propens))]
probs
probs }