Title: | Generalized Linear Mixed Models using Adaptive Gaussian Quadrature |
---|---|
Description: | Fits generalized linear mixed models for a single grouping factor under maximum likelihood approximating the integrals over the random effects with an adaptive Gaussian quadrature rule; Jose C. Pinheiro and Douglas M. Bates (1995) <doi:10.1080/10618600.1995.10474663>. |
Authors: | Dimitris Rizopoulos [aut, cre] |
Maintainer: | Dimitris Rizopoulos <[email protected]> |
License: | GPL (>= 3) |
Version: | 0.9-6 |
Built: | 2025-01-10 06:35:38 UTC |
Source: | https://github.com/drizopoulos/glmmadaptive |
Data set-up and calculation of marginal probabilities from a continuation ratio model
cr_setup(y, direction = c("forward", "backward")) cr_marg_probs(eta, direction = c("forward", "backward"))
cr_setup(y, direction = c("forward", "backward")) cr_marg_probs(eta, direction = c("forward", "backward"))
y |
a numeric vector denoting the ordinal response variable. |
direction |
character string specifying the direction of the continuation ratio
model; |
eta |
a numeric matrix of the linear predictor, with columns corresponding to the different levels of the ordinal response. |
Function cr_setup()
is based on the cr.setup()
function from package
rms.
Dimitris Rizopoulos [email protected]
Frank Harrell
n <- 300 # number of subjects K <- 8 # number of measurements per subject t_max <- 15 # maximum follow-up time # we constuct a data frame with the design: # everyone has a baseline measurment, and then measurements at random follow-up times DF <- data.frame(id = rep(seq_len(n), each = K), time = c(replicate(n, c(0, sort(runif(K - 1, 0, t_max))))), sex = rep(gl(2, n/2, labels = c("male", "female")), each = K)) # design matrices for the fixed and random effects X <- model.matrix(~ sex * time, data = DF)[, -1] Z <- model.matrix(~ 1, data = DF) thrs <- c(-1.5, 0, 0.9) # thresholds for the different ordinal categories betas <- c(-0.25, 0.24, -0.05) # fixed effects coefficients D11 <- 0.48 # variance of random intercepts D22 <- 0.1 # variance of random slopes # we simulate random effects b <- cbind(rnorm(n, sd = sqrt(D11)), rnorm(n, sd = sqrt(D22)))[, 1, drop = FALSE] # linear predictor eta_y <- drop(X %*% betas + rowSums(Z * b[DF$id, , drop = FALSE])) # linear predictor for each category eta_y <- outer(eta_y, thrs, "+") # marginal probabilities per category mprobs <- cr_marg_probs(eta_y) # we simulate ordinal longitudinal data DF$y <- unname(apply(mprobs, 1, sample, x = ncol(mprobs), size = 1, replace = TRUE)) # If you want to simulate from the backward formulation of the CR model, you need to # change `eta_y <- outer(eta_y, thrs, "+")` to `eta_y <- outer(eta_y, rev(thrs), "+")`, # and `mprobs <- cr_marg_probs(eta_y)` to `mprobs <- cr_marg_probs(eta_y, "backward")` ################################################# # prepare the data # If you want to fit the CR model under the backward formulation, you need to change # `cr_vals <- cr_setup(DF$y)` to `cr_vals <- cr_setup(DF$y, "backward")` cr_vals <- cr_setup(DF$y) cr_data <- DF[cr_vals$subs, ] cr_data$y_new <- cr_vals$y cr_data$cohort <- cr_vals$cohort # fit the model fm <- mixed_model(y_new ~ cohort + sex * time, random = ~ 1 | id, data = cr_data, family = binomial()) summary(fm)
n <- 300 # number of subjects K <- 8 # number of measurements per subject t_max <- 15 # maximum follow-up time # we constuct a data frame with the design: # everyone has a baseline measurment, and then measurements at random follow-up times DF <- data.frame(id = rep(seq_len(n), each = K), time = c(replicate(n, c(0, sort(runif(K - 1, 0, t_max))))), sex = rep(gl(2, n/2, labels = c("male", "female")), each = K)) # design matrices for the fixed and random effects X <- model.matrix(~ sex * time, data = DF)[, -1] Z <- model.matrix(~ 1, data = DF) thrs <- c(-1.5, 0, 0.9) # thresholds for the different ordinal categories betas <- c(-0.25, 0.24, -0.05) # fixed effects coefficients D11 <- 0.48 # variance of random intercepts D22 <- 0.1 # variance of random slopes # we simulate random effects b <- cbind(rnorm(n, sd = sqrt(D11)), rnorm(n, sd = sqrt(D22)))[, 1, drop = FALSE] # linear predictor eta_y <- drop(X %*% betas + rowSums(Z * b[DF$id, , drop = FALSE])) # linear predictor for each category eta_y <- outer(eta_y, thrs, "+") # marginal probabilities per category mprobs <- cr_marg_probs(eta_y) # we simulate ordinal longitudinal data DF$y <- unname(apply(mprobs, 1, sample, x = ncol(mprobs), size = 1, replace = TRUE)) # If you want to simulate from the backward formulation of the CR model, you need to # change `eta_y <- outer(eta_y, thrs, "+")` to `eta_y <- outer(eta_y, rev(thrs), "+")`, # and `mprobs <- cr_marg_probs(eta_y)` to `mprobs <- cr_marg_probs(eta_y, "backward")` ################################################# # prepare the data # If you want to fit the CR model under the backward formulation, you need to change # `cr_vals <- cr_setup(DF$y)` to `cr_vals <- cr_setup(DF$y, "backward")` cr_vals <- cr_setup(DF$y) cr_data <- DF[cr_vals$subs, ] cr_data$y_new <- cr_vals$y cr_data$cohort <- cr_vals$cohort # fit the model fm <- mixed_model(y_new ~ cohort + sex * time, random = ~ 1 | id, data = cr_data, family = binomial()) summary(fm)
Creates predicted values and their corresponding confidence interval for constructing an effects plot.
effectPlotData(object, newdata, level, ...) ## S3 method for class 'MixMod' effectPlotData(object, newdata, level = 0.95, marginal = FALSE, CR_cohort_varname = NULL, direction = NULL, K = 200, seed = 1, sandwich = FALSE, ...)
effectPlotData(object, newdata, level, ...) ## S3 method for class 'MixMod' effectPlotData(object, newdata, level = 0.95, marginal = FALSE, CR_cohort_varname = NULL, direction = NULL, K = 200, seed = 1, sandwich = FALSE, ...)
object |
an object inheriting from class |
newdata |
a data frame base on which predictions will be calculated. |
level |
a numeric scalar denoting the level of the confidence interval. |
marginal |
logical; if |
CR_cohort_varname |
a character string denoting the name of the cohort variable when a continuation ratio model is fitted. |
direction |
the |
K |
numeric scalar denoting the number of Monte Carlo samples from the approximate posterior of the parameters; applicable only for zero-inflated models. |
seed |
numerical scalar giving the seed to be used in the Monte Carlo scheme. |
sandwich |
logical; if |
... |
additional arguments passed to |
The confidence interval is calculated based on a normal approximation.
The data frame newdata
with extra columns pred
, low
and upp
.
Dimitris Rizopoulos [email protected]
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ # Fit a model fm1 <- mixed_model(fixed = y ~ year * group, random = ~ year | id, data = DF, family = binomial()) # An effects plot for the mean subject (i.e., with random effects equal to 0) nDF <- with(DF, expand.grid(year = seq(min(year), max(year), length.out = 15), group = levels(group))) plot_data <- effectPlotData(fm1, nDF) require("lattice") xyplot(pred + low + upp ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "log odds") expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") # we put the two groups in the same panel my.panel.bands <- function(x, y, upper, lower, fill, col, subscripts, ..., font, fontface) { upper <- upper[subscripts] lower <- lower[subscripts] panel.polygon(c(x, rev(x)), c(upper, rev(lower)), col = fill, border = FALSE, ...) } xyplot(expit(pred) ~ year, group = group, data = plot_data, upper = expit(plot_data$upp), low = expit(plot_data$low), type = "l", col = c("blue", "red"), fill = c("#0000FF80", "#FF000080"), panel = function (x, y, ...) { panel.superpose(x, y, panel.groups = my.panel.bands, ...) panel.xyplot(x, y, lwd = 2, ...) }, xlab = "Follow-up time", ylab = "Probabilities") # An effects plots for the marginal probabilities plot_data_m <- effectPlotData(fm1, nDF, marginal = TRUE, cores = 1L) expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data_m, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities")
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ # Fit a model fm1 <- mixed_model(fixed = y ~ year * group, random = ~ year | id, data = DF, family = binomial()) # An effects plot for the mean subject (i.e., with random effects equal to 0) nDF <- with(DF, expand.grid(year = seq(min(year), max(year), length.out = 15), group = levels(group))) plot_data <- effectPlotData(fm1, nDF) require("lattice") xyplot(pred + low + upp ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "log odds") expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") # we put the two groups in the same panel my.panel.bands <- function(x, y, upper, lower, fill, col, subscripts, ..., font, fontface) { upper <- upper[subscripts] lower <- lower[subscripts] panel.polygon(c(x, rev(x)), c(upper, rev(lower)), col = fill, border = FALSE, ...) } xyplot(expit(pred) ~ year, group = group, data = plot_data, upper = expit(plot_data$upp), low = expit(plot_data$low), type = "l", col = c("blue", "red"), fill = c("#0000FF80", "#FF000080"), panel = function (x, y, ...) { panel.superpose(x, y, panel.groups = my.panel.bands, ...) panel.xyplot(x, y, lwd = 2, ...) }, xlab = "Follow-up time", ylab = "Probabilities") # An effects plots for the marginal probabilities plot_data_m <- effectPlotData(fm1, nDF, marginal = TRUE, cores = 1L) expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data_m, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities")
Specifies the information required to fit a Beta, zero-inflated and hurdle Poisson,
zero-inflated and hurdle Negative Binomial, a hurdle normal and a hurdle Beta
mixed-effects model, using
mixed_model()
.
students.t(df = stop("'df' must be specified"), link = "identity") beta.fam() zi.poisson() zi.binomial() zi.negative.binomial() hurdle.poisson() hurdle.negative.binomial() hurdle.lognormal() hurdle.beta.fam() unit.lindley() beta.binomial(link = "logit") Gamma.fam() censored.normal()
students.t(df = stop("'df' must be specified"), link = "identity") beta.fam() zi.poisson() zi.binomial() zi.negative.binomial() hurdle.poisson() hurdle.negative.binomial() hurdle.lognormal() hurdle.beta.fam() unit.lindley() beta.binomial(link = "logit") Gamma.fam() censored.normal()
link |
name of the link function. |
df |
the degrees of freedom of the Student's t distribution. |
Currently only the log-link is implemented for the Poisson, negative binomial and Gamma models, the logit link for the beta and hurdle beta models and the identity link for the log-normal model.
# simulate some data from a negative binomial model set.seed(102) dd <- expand.grid(f1 = factor(1:3), f2 = LETTERS[1:2], g = 1:30, rep = 1:15, KEEP.OUT.ATTRS = FALSE) mu <- 5*(-4 + with(dd, as.integer(f1) + 4 * as.numeric(f2))) dd$y <- rnbinom(nrow(dd), mu = mu, size = 0.5) # Fit a zero-inflated Poisson model, with only fixed effects in the # zero-inflated part fm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.poisson(), zi_fixed = ~ 1) summary(fm1) # We extend the previous model allowing also for a random intercept in the # zero-inflated part fm2 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.poisson(), zi_fixed = ~ 1, zi_random = ~ 1 | g) # We do a likelihood ratio test between the two models anova(fm1, fm2) ############################################################################# ############################################################################# # The same as above but with a negative binomial model gm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.negative.binomial(), zi_fixed = ~ 1) summary(gm1) # We do a likelihood ratio test between the Poisson and negative binomial models anova(fm1, gm1)
# simulate some data from a negative binomial model set.seed(102) dd <- expand.grid(f1 = factor(1:3), f2 = LETTERS[1:2], g = 1:30, rep = 1:15, KEEP.OUT.ATTRS = FALSE) mu <- 5*(-4 + with(dd, as.integer(f1) + 4 * as.numeric(f2))) dd$y <- rnbinom(nrow(dd), mu = mu, size = 0.5) # Fit a zero-inflated Poisson model, with only fixed effects in the # zero-inflated part fm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.poisson(), zi_fixed = ~ 1) summary(fm1) # We extend the previous model allowing also for a random intercept in the # zero-inflated part fm2 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.poisson(), zi_fixed = ~ 1, zi_random = ~ 1 | g) # We do a likelihood ratio test between the two models anova(fm1, fm2) ############################################################################# ############################################################################# # The same as above but with a negative binomial model gm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = zi.negative.binomial(), zi_fixed = ~ 1) summary(gm1) # We do a likelihood ratio test between the Poisson and negative binomial models anova(fm1, gm1)
This package fits generalized linear mixed models for a single grouping factor under maximum likelihood approximating the integrals over the random effects with an adaptive Gaussian quadrature rule.
Package: | GLMMadaptive |
Type: | Package |
Version: | 0.9-5 |
Date: | 2024-06-17 |
License: | GPL (>=3) |
This package fits mixed effects models for grouped / repeated measurements data for which the integral over the random effects in the definition of the marginal likelihood cannot be solved analytically. The package approximates these integrals using the adaptive Gauss-Hermite quadrature rule.
Multiple random effects terms can be included for the grouping factor (e.g., random intercepts, random linear slopes, random quadratic slopes), but currently only a single grouping factor is allowed.
The package also offers several utility functions that can extract useful information from fitted mixed effects models. The most important of those are included in the See also Section below.
Dimitris Rizopoulos
Maintainer: Dimitris Rizopoulos <[email protected]>
Fitzmaurice, G., Laird, N. and Ware J. (2011). Applied Longitudinal Analysis, 2nd Ed. New York: John Wiley & Sons.
Molenberghs, G. and Verbeke, G. (2005). Models for Discrete Longitudinal Data. New York: Springer-Verlag.
mixed_model
,
methods.MixMod
,
effectPlotData
,
marginal_coefs
Calculates marginal coefficients and their standard errors from fitted generalized linear mixed models.
marginal_coefs(object, ...) ## S3 method for class 'MixMod' marginal_coefs(object, std_errors = FALSE, link_fun = NULL, M = 3000, K = 100, seed = 1, cores = max(parallel::detectCores() - 1, 1), sandwich = FALSE, ...)
marginal_coefs(object, ...) ## S3 method for class 'MixMod' marginal_coefs(object, std_errors = FALSE, link_fun = NULL, M = 3000, K = 100, seed = 1, cores = max(parallel::detectCores() - 1, 1), sandwich = FALSE, ...)
object |
an object inheriting from class |
std_errors |
logical indicating whether standard errors are to be computed. |
link_fun |
a function transforming the mean of the repeated measurements outcome to the
linear predictor scale. Typically, this derived from the |
M |
numeric scalar denoting the number of Monte Carlo samples. |
K |
numeric scalar denoting the number of samples from the sampling distribution of the maximum likelihood estimates. |
seed |
integer denoting the seed for the random number generation. |
cores |
integer giving the number of cores to use; applicable only when
|
sandwich |
logical; if |
... |
extra arguments; currently none is used. |
It uses the approach of Hedeker et al. (2017) to calculate marginal coefficients from
mixed models with nonlinear link functions. The marginal probabilities are calculated
using Monte Carlo integration over the random effects with M
samples, by sampling
from the estimated prior distribution, i.e., a multivariate normal distribution with mean
0 and covariance matrix , where
denotes the estimated
covariance matrix of the random effects.
To calculate the standard errors, the Monte Carlo integration procedure is repeated
K
times, where each time instead of the maximum likelihood estimates of the fixed
effects and the covariance matrix of the random effects, a realization is used from the
sampling distribution of the maximum likelihood estimates. To speed-up this process,
package parallel is used.
A list of class "m_coefs"
with components betas
the marginal coefficients,
and when std_errors = TRUE
, the extra components var_betas
the estimated
covariance matrix of the marginal coefficients, and coef_table
a numeric matrix
with the estimated marginal coefficients, their standard errors and corresponding
p-values using the normal approximation.
Dimitris Rizopoulos [email protected]
Hedeker, D., du Toit, S. H., Demirtas, H. and Gibbons, R. D. (2018), A note on marginalization of regression parameters from mixed models of binary outcomes. Biometrics 74, 354–361. doi:10.1111/biom.12707
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year * group, random = ~ 1 | id, data = DF, family = binomial()) fixef(fm1) marginal_coefs(fm1) marginal_coefs(fm1, std_errors = TRUE, cores = 1L)
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year * group, random = ~ 1 | id, data = DF, family = binomial()) fixef(fm1) marginal_coefs(fm1) marginal_coefs(fm1, std_errors = TRUE, cores = 1L)
Fits generalized linear mixed effects models under maximum likelihood using adaptive Gaussian quadrature.
mixed_model(fixed, random, data, family, weights = NULL, na.action = na.exclude, zi_fixed = NULL, zi_random = NULL, penalized = FALSE, n_phis = NULL, initial_values = NULL, control = list(), ...)
mixed_model(fixed, random, data, family, weights = NULL, na.action = na.exclude, zi_fixed = NULL, zi_random = NULL, penalized = FALSE, n_phis = NULL, initial_values = NULL, control = list(), ...)
fixed |
a formula for the fixed-effects part of the model, including the outcome. |
random |
a formula for the random-effects part of the model. This should only contain
the right-hand side part, e.g., |
data |
a data.frame containing the variables required in |
family |
a |
weights |
a numeric vector of weights. These are simple multipliers on the log-likelihood contributions of each group/cluster, i.e., we presume that there are multiple replicates of each group/cluster denoted by the weights. The length of 'weights' need to be equal to the number of independent groups/clusters in the data. |
na.action |
what to do with missing values in |
zi_fixed , zi_random
|
formulas for the fixed and random effects of the zero inflated part. |
penalized |
logical or a list. If logical and equal to |
n_phis |
a numeric scalar; in case the family corresponds to a distribution that has extra (dispersion/shape) parameters, you need to specify how many extra parameters are needed. |
initial_values |
a list of initial values. This can have up to three components, namely,
|
control |
a list with the following components:
|
... |
arguments passed to |
General: The mixed_model()
function fits mixed effects models in which the
integrals over the random effects in the definition of the marginal log-likelihood cannot
be solved analytically and need to be approximated. The function works under the
assumption of normally distributed random effects with mean zero and variance-covariance
matrix . These integrals are approximated numerically using an adaptive
Gauss-Hermite quadrature rule. Using the control argument
nAGQ
, the user can
specify the number of quadrature points used in the approximation.
User-defined family: The user can define its own family object; for an example,
see the help page of negative.binomial
.
Optimization: A hybrid approach is used, starting with iter_EM
iterations
and unless convergence was achieved it continuous with a direct optimization of the
log-likelihood using function optim
and the algorithm specified by
optim_method
. For stability and speed, the derivative of the log-likelihood with
respect to the parameters are internally programmed.
An object of class "MixMod"
with components:
coefficients |
a numeric vector with the estimated fixed effects. |
phis |
a numeric vector with the estimated extra parameters. |
D |
a numeric matrix denoting the estimated covariance matrix of the random effects. |
post_modes |
a numeric matrix with the empirical Bayes estimates of the random effects. |
post_vars |
a list of numeric matrices with the posterior variances of the random effects. |
logLik |
a numeric scalar denoting the log-likelihood value at the end of the optimization procedure. |
Hessian |
a numeric matrix denoting the Hessian matrix at the end of the optimization procedure. |
converged |
a logical indicating whether convergence was attained. |
data |
a copy of the |
id |
a copy of the grouping variable from |
id_name |
a character string with the name of the grouping variable. |
Terms |
a list with two terms components, |
model_frames |
a list with two model.frame components, |
control |
a copy of the (user-specific) |
Funs |
a list of functions used in the optimization procedure. |
family |
a copy of the |
call |
the matched call. |
Dimitris Rizopoulos [email protected]
methods.MixMod
,
effectPlotData
,
marginal_coefs
# simulate some data set.seed(123L) n <- 200 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year * group, random = ~ 1 | id, data = DF, family = binomial()) # fixed effects fixef(fm1) # random effects head(ranef(fm1)) # detailed output summary(fm1) # fitted values for the 'mean subject', i.e., with # random effects values equal to 0 head(fitted(fm1, type = "mean_subject")) # fitted values for the conditioning on the estimated random effects head(fitted(fm1, type = "subject_specific")) ############## fm2 <- mixed_model(fixed = y ~ year, random = ~ 1 | id, data = DF, family = binomial()) # likelihood ratio test between the two models anova(fm2, fm1) # the same hypothesis but with a Wald test anova(fm1, L = rbind(c(0, 0, 1, 0), c(0, 0, 0, 1))) ############## # An effects plot for the mean subject (i.e., with random effects equal to 0) nDF <- with(DF, expand.grid(year = seq(min(year), max(year), length.out = 15), group = levels(group))) plot_data <- effectPlotData(fm2, nDF) require("lattice") xyplot(pred + low + upp ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "log odds") expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") # An effects plots for the marginal probabilities plot_data_m <- effectPlotData(fm2, nDF, marginal = TRUE, cores = 1L) expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data_m, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") ############## # include random slopes fm1_slp <- update(fm1, random = ~ year | id) # increase the number of quadrature points to 15 fm1_slp_q15 <- update(fm1_slp, nAGQ = 15) # a diagonal covariance matrix for the random effects fm1_slp_diag <- update(fm1, random = ~ year || id) anova(fm1_slp_diag, fm1_slp)
# simulate some data set.seed(123L) n <- 200 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year * group, random = ~ 1 | id, data = DF, family = binomial()) # fixed effects fixef(fm1) # random effects head(ranef(fm1)) # detailed output summary(fm1) # fitted values for the 'mean subject', i.e., with # random effects values equal to 0 head(fitted(fm1, type = "mean_subject")) # fitted values for the conditioning on the estimated random effects head(fitted(fm1, type = "subject_specific")) ############## fm2 <- mixed_model(fixed = y ~ year, random = ~ 1 | id, data = DF, family = binomial()) # likelihood ratio test between the two models anova(fm2, fm1) # the same hypothesis but with a Wald test anova(fm1, L = rbind(c(0, 0, 1, 0), c(0, 0, 0, 1))) ############## # An effects plot for the mean subject (i.e., with random effects equal to 0) nDF <- with(DF, expand.grid(year = seq(min(year), max(year), length.out = 15), group = levels(group))) plot_data <- effectPlotData(fm2, nDF) require("lattice") xyplot(pred + low + upp ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "log odds") expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") # An effects plots for the marginal probabilities plot_data_m <- effectPlotData(fm2, nDF, marginal = TRUE, cores = 1L) expit <- function (x) exp(x) / (1 + exp(x)) xyplot(expit(pred) + expit(low) + expit(upp) ~ year | group, data = plot_data_m, type = "l", lty = c(1, 2, 2), col = c(2, 1, 1), lwd = 2, xlab = "Follow-up time", ylab = "Probabilities") ############## # include random slopes fm1_slp <- update(fm1, random = ~ year | id) # increase the number of quadrature points to 15 fm1_slp_q15 <- update(fm1_slp, nAGQ = 15) # a diagonal covariance matrix for the random effects fm1_slp_diag <- update(fm1, random = ~ year || id) anova(fm1_slp_diag, fm1_slp)
Methods for object of class "MixMod"
for standard generic functions.
coef(object, ...) ## S3 method for class 'MixMod' coef(object, sub_model = c("main", "zero_part"), ...) fixef(object, ...) ## S3 method for class 'MixMod' fixef(object, sub_model = c("main", "zero_part"), ...) ranef(object, ...) ## S3 method for class 'MixMod' ranef(object, post_vars = FALSE, ...) confint(object, parm, level = 0.95, ...) ## S3 method for class 'MixMod' confint(object, parm = c("fixed-effects", "var-cov","extra", "zero_part"), level = 0.95, sandwich = FALSE, ...) anova(object, ...) ## S3 method for class 'MixMod' anova(object, object2, test = TRUE, L = NULL, sandwich = FALSE, ...) fitted(object, ...) ## S3 method for class 'MixMod' fitted(object, type = c("mean_subject", "subject_specific", "marginal"), link_fun = NULL, ...) residuals(object, ...) ## S3 method for class 'MixMod' residuals(object, type = c("mean_subject", "subject_specific", "marginal"), link_fun = NULL, tasnf_y = function (x) x, ...) predict(object, ...) ## S3 method for class 'MixMod' predict(object, newdata, newdata2 = NULL, type_pred = c("response", "link"), type = c("mean_subject", "subject_specific", "marginal", "zero_part"), se.fit = FALSE, M = 300, df = 10, scale = 0.3, level = 0.95, seed = 1, return_newdata = FALSE, sandwich = FALSE, ...) simulate(object, nsim = 1, seed = NULL, ...) ## S3 method for class 'MixMod' simulate(object, nsim = 1, seed = NULL, type = c("subject_specific", "mean_subject"), new_RE = FALSE, acount_MLEs_var = FALSE, sim_fun = NULL, sandwich = FALSE, ...) terms(x, ...) ## S3 method for class 'MixMod' terms(x, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) formula(x, ...) ## S3 method for class 'MixMod' formula(x, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) model.frame(formula, ...) ## S3 method for class 'MixMod' model.frame(formula, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) model.matrix(object, ...) ## S3 method for class 'MixMod' model.matrix(object, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) nobs(object, ...) ## S3 method for class 'MixMod' nobs(object, level = 1, ...) VIF(object, ...) ## S3 method for class 'MixMod' VIF(object, type = c("fixed", "zi_fixed"), ...) cooks.distance(model, ...) ## S3 method for class 'MixMod' cooks.distance(model, cores = max(parallel::detectCores() - 1, 1), ...)
coef(object, ...) ## S3 method for class 'MixMod' coef(object, sub_model = c("main", "zero_part"), ...) fixef(object, ...) ## S3 method for class 'MixMod' fixef(object, sub_model = c("main", "zero_part"), ...) ranef(object, ...) ## S3 method for class 'MixMod' ranef(object, post_vars = FALSE, ...) confint(object, parm, level = 0.95, ...) ## S3 method for class 'MixMod' confint(object, parm = c("fixed-effects", "var-cov","extra", "zero_part"), level = 0.95, sandwich = FALSE, ...) anova(object, ...) ## S3 method for class 'MixMod' anova(object, object2, test = TRUE, L = NULL, sandwich = FALSE, ...) fitted(object, ...) ## S3 method for class 'MixMod' fitted(object, type = c("mean_subject", "subject_specific", "marginal"), link_fun = NULL, ...) residuals(object, ...) ## S3 method for class 'MixMod' residuals(object, type = c("mean_subject", "subject_specific", "marginal"), link_fun = NULL, tasnf_y = function (x) x, ...) predict(object, ...) ## S3 method for class 'MixMod' predict(object, newdata, newdata2 = NULL, type_pred = c("response", "link"), type = c("mean_subject", "subject_specific", "marginal", "zero_part"), se.fit = FALSE, M = 300, df = 10, scale = 0.3, level = 0.95, seed = 1, return_newdata = FALSE, sandwich = FALSE, ...) simulate(object, nsim = 1, seed = NULL, ...) ## S3 method for class 'MixMod' simulate(object, nsim = 1, seed = NULL, type = c("subject_specific", "mean_subject"), new_RE = FALSE, acount_MLEs_var = FALSE, sim_fun = NULL, sandwich = FALSE, ...) terms(x, ...) ## S3 method for class 'MixMod' terms(x, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) formula(x, ...) ## S3 method for class 'MixMod' formula(x, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) model.frame(formula, ...) ## S3 method for class 'MixMod' model.frame(formula, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) model.matrix(object, ...) ## S3 method for class 'MixMod' model.matrix(object, type = c("fixed", "random", "zi_fixed", "zi_random"), ...) nobs(object, ...) ## S3 method for class 'MixMod' nobs(object, level = 1, ...) VIF(object, ...) ## S3 method for class 'MixMod' VIF(object, type = c("fixed", "zi_fixed"), ...) cooks.distance(model, ...) ## S3 method for class 'MixMod' cooks.distance(model, cores = max(parallel::detectCores() - 1, 1), ...)
object , object2 , x , formula , model
|
objects inheriting from class |
sub_model |
character string indicating for which sub-model to extract the estimated coefficients; it is only relevant for zero-inflated models. |
post_vars |
logical; if |
parm |
character string; for which type of parameters to calculate confidence
intervals. Option |
level |
numeric scalar between 0 and 1 denoting the level of the confidence interval.
In the |
test |
logical; should a p-value be calculated. |
L |
a numeric matrix representing a contrasts matrix. This is only used when in
|
sandwich |
logical; if |
type |
character string indicating the type of fitted values / residuals / predictions
/ variance inflation factors to calculate. Option |
link_fun |
the |
tasnf_y |
a function to transform the grouped / repeated measurements outcome before calculating the residuals; for example, relevant in two-part models for semi-continuous data, in which it is assumed that the log outcome follows a normal distribution. |
newdata , newdata2
|
a data frame based on which predictions are to be calculated.
|
type_pred |
character string indicating at which scale to calculate predictions.
Options are |
se.fit |
logical, if |
M |
numeric scalar denoting the number of Monte Carlo samples; see Details for more information. |
df |
numeric scalar denoting the degrees of freedom for the Student's t proposal distribution; see Details for more information. |
scale |
numeric scalar or vector denoting the scaling applied to the subject-specific covariance matrices of the random effects; see Details for more information. |
seed |
numerical scalar giving the seed to be used in the Monte Carlo scheme. |
return_newdata |
logical; if |
nsim |
numeric scalar giving the number of times to simulate the response variable. |
new_RE |
logical; if |
acount_MLEs_var |
logical; if |
sim_fun |
a function based on which to simulate the response variable. This is
relevant for non-standard models. The |
cores |
the number of cores to use in the computation. |
... |
further arguments; currently none is used. |
In generic terms, we assume that the mean of the outcome (
denotes the subjects) conditional on the random effects is given by the equation:
where denotes the link function,
the vector of random effects,
the vector of fixed effects, and
and
the design matrices
for the fixed and random effects, respectively.
Argument type_pred
of predict()
specifies whether predictions will be
calculated in the link / linear predictor scale, i.e., or in the response
scale, i.e.,
.
When type = "mean_subject"
, predictions are calculated using only the fixed
effects, i.e., the part, where
is evaluated in
newdata
.
This corresponds to predictions for the 'mean' subjects, i.e., subjects who have
random effects value equal to 0. Note, that in the case of nonlinear link functions this
does not correspond to the averaged over the population predictions (i.e., marginal
predictions).
When type = "marginal"
, predictions are calculated using only the fixed
effects, i.e., the part, where
is evaluated in
newdata
,
but with coefficients the marginalized coefficients obtain from
marginal_coefs
. These predictions will be marginal / population averaged
predictions.
When type = "zero_part"
, predictions are calculated for the logistic regression of
the extra zero-part of the model (i.e., applicable for zero-inflated and hurdle models).
When type = "subject_specific"
, predictions are calculated using both the fixed-
and random-effects parts, i.e., , where
and
are evaluated in
newdata
. Estimates for the random effects of each subject are
obtained as modes from the posterior distribution evaluated in
newdata
and with (denoting the parameters of the model, fixed effects
and variance components) replaced by their maximum likelihood estimates.
Notes: (i) When se.fit = TRUE
and type_pred = "response"
, the
standard errors returned are on the linear predictor scale, not the response scale.
(ii) When se.fit = TRUE
and the model contains an extra zero-part, no standard
errors are computed when type = "mean_subject"
. (iii) When the model contains an
extra zero-part, type = "marginal"
predictions are not yet implemented.
When se.fit = TRUE
and type = "subject_specific"
, standard errors and
confidence intervals for the subject-specific predictions are obtained by a Monte Carlo
scheme entailing three steps repeated M
times, namely
Account for the variability of maximum likelihood estimates (MLES) by
simulating a new value for the parameters
from a
multivariate normal distribution with mean the MLEs and covariance matrix the
covariance matrix of the MLEs.
Account for the variability in the random effects estimates by
simulating a new value for the random effects
from the posterior
distribution
. Because the posterior distribution does not
have a closed-form, a Metropolis-Hastings algorithm is used to sample the new value
using as proposal distribution a multivariate Student's-t distribution
with degrees of freedom
df
, centered at the mode of the posterior distribution
with
the MLEs, and scale matrix the inverse
Hessian matrix of the log density of
evaluated at the modes,
but multiplied by
scale
. The scale
and df
parameters can be used
to adjust the acceptance rate.
The predictions are calculated using .
Argument newdata2
can be used to calculate dynamic subject-specific predictions.
I.e., using the observed responses in
newdata
, estimates of the random
effects of each subject are obtained. For the same subjects we want to obtain predictions
in new covariates settings for which no response data are yet available. For example,
in a longitudinal study, for a subject we have responses up to a follow-up
(
newdata
) and we want the prediction at (
newdata2
).
The estimated fixed and random effects, coefficients (this is similar as in package nlme), confidence intervals fitted values (on the scale on the response) and residuals.
Dimitris Rizopoulos [email protected]
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year + group, random = ~ year | id, data = DF, family = binomial()) head(coef(fm1)) fixef(fm1) head(ranef(fm1)) confint(fm1) confint(fm1, "var-cov") head(fitted(fm1, "subject_specific")) head(residuals(fm1, "marginal")) fm2 <- mixed_model(fixed = y ~ year * group, random = ~ year | id, data = DF, family = binomial()) # likelihood ratio test between fm1 and fm2 anova(fm1, fm2) # the same but with a Wald test anova(fm2, L = rbind(c(0, 0, 0, 1)))
# simulate some data set.seed(123L) n <- 500 K <- 15 t.max <- 25 betas <- c(-2.13, -0.25, 0.24, -0.05) D <- matrix(0, 2, 2) D[1:2, 1:2] <- c(0.48, -0.08, -0.08, 0.18) times <- c(replicate(n, c(0, sort(runif(K-1, 0, t.max))))) group <- sample(rep(0:1, each = n/2)) DF <- data.frame(year = times, group = factor(rep(group, each = K))) X <- model.matrix(~ group * year, data = DF) Z <- model.matrix(~ year, data = DF) b <- cbind(rnorm(n, sd = sqrt(D[1, 1])), rnorm(n, sd = sqrt(D[2, 2]))) id <- rep(1:n, each = K) eta.y <- as.vector(X %*% betas + rowSums(Z * b[id, ])) DF$y <- rbinom(n * K, 1, plogis(eta.y)) DF$id <- factor(id) ################################################ fm1 <- mixed_model(fixed = y ~ year + group, random = ~ year | id, data = DF, family = binomial()) head(coef(fm1)) fixef(fm1) head(ranef(fm1)) confint(fm1) confint(fm1, "var-cov") head(fitted(fm1, "subject_specific")) head(residuals(fm1, "marginal")) fm2 <- mixed_model(fixed = y ~ year * group, random = ~ year | id, data = DF, family = binomial()) # likelihood ratio test between fm1 and fm2 anova(fm1, fm2) # the same but with a Wald test anova(fm2, L = rbind(c(0, 0, 0, 1)))
Specifies the information required to fit a Negative Binomial generalized linear mixed
model, using mixed_model()
.
negative.binomial()
negative.binomial()
Currently only the log-link is implemented.
Dimitris Rizopoulos [email protected]
# simulate some data set.seed(102) dd <- expand.grid(f1 = factor(1:3), f2 = LETTERS[1:2], g = 1:30, rep = 1:15, KEEP.OUT.ATTRS = FALSE) mu <- 5 * (-4 + with(dd, as.integer(f1) + 4 * as.numeric(f2))) dd$y <- rnbinom(nrow(dd), mu = mu, size = 0.5) gm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = negative.binomial()) summary(gm1) # We do a likelihood ratio test with the Poisson mixed model gm0 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = poisson()) anova(gm0, gm1) # Define a cutom-made family function to be used with mixed_model() # the required components are 'family', 'link', 'linkfun', 'linkinv' and 'log_dens'; # the extra functions 'score_eta_fun' and 'score_phis_fun' can be skipped and will # internally approximated using numeric derivatives (though it is better that you provide # them). my_negBinom <- function (link = "log") { stats <- make.link(link) log_dens <- function (y, eta, mu_fun, phis, eta_zi) { # the log density function phis <- exp(phis) mu <- mu_fun(eta) log_mu_phis <- log(mu + phis) comp1 <- lgamma(y + phis) - lgamma(phis) - lgamma(y + 1) comp2 <- phis * log(phis) - phis * log_mu_phis comp3 <- y * log(mu) - y * log_mu_phis out <- comp1 + comp2 + comp3 attr(out, "mu_y") <- mu out } score_eta_fun <- function (y, mu, phis, eta_zi) { # the derivative of the log density w.r.t. mu phis <- exp(phis) mu_phis <- mu + phis comp2 <- - phis / mu_phis comp3 <- y / mu - y / mu_phis # the derivative of mu w.r.t. eta (this depends on the chosen link function) mu.eta <- mu (comp2 + comp3) * mu.eta } score_phis_fun <- function (y, mu, phis, eta_zi) { # the derivative of the log density w.r.t. phis phis <- exp(phis) mu_phis <- mu + phis comp1 <- digamma(y + phis) - digamma(phis) comp2 <- log(phis) + 1 - log(mu_phis) - phis / mu_phis comp3 <- - y / mu_phis comp1 + comp2 + comp3 } structure(list(family = "user Neg Binom", link = stats$name, linkfun = stats$linkfun, linkinv = stats$linkinv, log_dens = log_dens, score_eta_fun = score_eta_fun, score_phis_fun = score_phis_fun), class = "family") } fm <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = my_negBinom(), n_phis = 1, initial_values = list("betas" = poisson())) summary(fm)
# simulate some data set.seed(102) dd <- expand.grid(f1 = factor(1:3), f2 = LETTERS[1:2], g = 1:30, rep = 1:15, KEEP.OUT.ATTRS = FALSE) mu <- 5 * (-4 + with(dd, as.integer(f1) + 4 * as.numeric(f2))) dd$y <- rnbinom(nrow(dd), mu = mu, size = 0.5) gm1 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = negative.binomial()) summary(gm1) # We do a likelihood ratio test with the Poisson mixed model gm0 <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = poisson()) anova(gm0, gm1) # Define a cutom-made family function to be used with mixed_model() # the required components are 'family', 'link', 'linkfun', 'linkinv' and 'log_dens'; # the extra functions 'score_eta_fun' and 'score_phis_fun' can be skipped and will # internally approximated using numeric derivatives (though it is better that you provide # them). my_negBinom <- function (link = "log") { stats <- make.link(link) log_dens <- function (y, eta, mu_fun, phis, eta_zi) { # the log density function phis <- exp(phis) mu <- mu_fun(eta) log_mu_phis <- log(mu + phis) comp1 <- lgamma(y + phis) - lgamma(phis) - lgamma(y + 1) comp2 <- phis * log(phis) - phis * log_mu_phis comp3 <- y * log(mu) - y * log_mu_phis out <- comp1 + comp2 + comp3 attr(out, "mu_y") <- mu out } score_eta_fun <- function (y, mu, phis, eta_zi) { # the derivative of the log density w.r.t. mu phis <- exp(phis) mu_phis <- mu + phis comp2 <- - phis / mu_phis comp3 <- y / mu - y / mu_phis # the derivative of mu w.r.t. eta (this depends on the chosen link function) mu.eta <- mu (comp2 + comp3) * mu.eta } score_phis_fun <- function (y, mu, phis, eta_zi) { # the derivative of the log density w.r.t. phis phis <- exp(phis) mu_phis <- mu + phis comp1 <- digamma(y + phis) - digamma(phis) comp2 <- log(phis) + 1 - log(mu_phis) - phis / mu_phis comp3 <- - y / mu_phis comp1 + comp2 + comp3 } structure(list(family = "user Neg Binom", link = stats$name, linkfun = stats$linkfun, linkinv = stats$linkinv, log_dens = log_dens, score_eta_fun = score_eta_fun, score_phis_fun = score_phis_fun), class = "family") } fm <- mixed_model(fixed = y ~ f1 * f2, random = ~ 1 | g, data = dd, family = my_negBinom(), n_phis = 1, initial_values = list("betas" = poisson())) summary(fm)
Calculates the logarithmic, quadratic/Brier and spherical based on a fitted mixed model for categorical data.
scoring_rules(object, newdata, newdata2 = NULL, max_count = 2000, return_newdata = FALSE)
scoring_rules(object, newdata, newdata2 = NULL, max_count = 2000, return_newdata = FALSE)
object |
an object inheriting from class |
newdata |
a data.frame based on which to estimate the random effect and calculate predictions. It should contain the response variable. |
newdata2 |
a data.frame based on which to estimate the random effect and calculate predictions. It should contain the response variable. |
max_count |
numeric scalar denoting the maximum count up to which to calculate probabilities; this is relevant for count response data. |
return_newdata |
logical; if |
A data.frame with (extra) columns the values of the logarithmic, quadratic and spherical
scoring rules calculated based on the fitted model and the observed responses in
newdata
or newdata2
.
Dimitris Rizopoulos [email protected]
Carvalho, A. (2016). An overview of applications of proper scoring rules. Decision Analysis 13, 223–242. doi:10.1287/deca.2016.0337
NA NA NA
NA NA NA