set.seed(1288)
library(patchwork) # for creating multi-panel plots
library(ggplot2) # for plotting
library(R2jags) # for interfacing with JAGS
library(MCMCvis) # for summarizing MCMC output
library(mcmcplots) # for visualizing MCMC output
library(sjPlot) # for creating tables
library(modelsummary) # for creating tables
library(dplyr) # for data wrangling
library(knitr) # for creating reproducible reports
library(MASS) # for fitting negative binomial regression models
library(pscl) # for fitting zero-inflated models
library(kableExtra) # for tables
library(tinytable) # for tables
17 Models for zero-inflated data
Learning objectives
Be able to fit models to response data with lots of zeros (hurdle and zero-inflated models).
Be able to describe these models and their assumptions using equations and text and match parameters in these equations to estimates in computer output.
17.1 R packages
We begin by loading a few packages upfront:
In addition, we will use data and functions from the following packages:
Data4Ecologists
for the slugs data setglmmTMB
for fitting zero-inflated models
17.2 Zero-inflation
Presence-absence and count data collected in ecological monitoring efforts are often analyzed using models that allow for zero-inflation – i.e., an overabundance of 0’s. The first thing to note about zero-inflation is that it refers to response data, \(Y_i\), not predictors, \(X_i\). Although we will focus on zero-inflated count data, zero-inflation is also relevant to binary data (e.g., occupancy models as discussed in Chapter 20 of Kéry 2010), and continuous data (e.g., Friederichs et al. 2011).
There are many reasons why we may end up with an overabundance of zero counts in ecological monitoring data sets:
- sites may not be suitable for the species;
- the species may occur at low densities, and therefore suitable sites may not all be occupied;
- our sampling effort may not have been sufficient to detect individuals that use the site or we may have sampled during times when individuals are difficult to detect;
- we may have mistaken individuals for the wrong species.
Common statistical distributions used to model count data, including the Poisson and Negative Binomial distributions, allow for zeros:
- Poisson Distribution: \(P(Y_i = 0) = \exp(-\lambda_i)\)
- Negative Binomial Distribution: \(P(Y_i = 0) = \Big(\frac{\theta}{\mu+\theta}\Big)^\theta\)
Zero-inflation refers to situations where there are more zeros than expected given the assumed statistical distribution. In regression applications, we are allowing the mean of these distributions to depend on predictor variables. There may be some regions of “predictor space” where the mean is really small, and thus, where we might expect to find many zeros. So, a lot of zeros does not necessarily mean that data are zero-inflated. As the mean gets larger, however, we should expect fewer zeros. When certain regions of predictor space have both a large mean and many zeros, we might want to consider models that allow for zero-inflation.
17.3 Testing for excess zeros
How can we determine if we have excess zeros (i.e., zero-inflated data)? One option would be to modify our goodness-of-fit test (see Section 15.4.5) to use the number of zeros in our data set as the test statistic (rather than the sum of Pearson residuals). To demonstrate, let’s revisit the slug data from Chapter 10 and Chapter 15.
library(Data4Ecologists)
data(slugs)
We will use the total number of zeros as our test statistic: \(T = \sum_{i=1}^n I(Y_i=0)\). Let’s work through the different steps involved in the hypothesis test:
- State null and alternative hypotheses,
- \(H_0:\) the data are consistent with a Poisson
- \(H_A:\) the data are not consistent with a model with separate \(\lambda\)s for the field and Rookery sites; there are more 0’s than expected if the data followed separate Poisson distributions for each site.
Calculate a statistic that measures the discrepancy between the data and the null hypothesis. Here, we will use \(T =\) the number of zeros in the data set.
Determine the distribution of the statistic in step 2 when the null hypothesis is true. We will determine the distribution of \(T\) when the null hypothesis is true using simulations, allowing for uncertainty in our values of \(\lambda_i\).
Compare the statistic for the observed data [step 2] to the distribution of the statistic when the null hypothesis is true [step 3]. The p-value = the probability of getting a value of \(T\) as big or bigger than the value for our observed data.
# Fitted model
<-glm(slugs~field, data=slugs, family=poisson())
fit.pois
# Number of simulations
<-10000
nsims
# Set up matrix to hold goodness-of-fit statistics
<-matrix(NA, nsims, 1)
zeros.sim<-nrow(slugs) # number of observations
nobs
# Extract the estimated coefficients and their asymptotic variance/covariance matrix
# Use these values to generate nsims new betas (beta.hat)
<-MASS::mvrnorm(nsims, coef(fit.pois), vcov(fit.pois))
beta.hat
# Design matrix for creating new lambda^'s
<-model.matrix(fit.pois)
xmatfor(i in 1:nsims){
# Generate lambda^'s
<-exp(xmat%*%beta.hat[i,])
lambda.hat# Generate new simulated values
<-rpois(nobs, lambda = lambda.hat)
new.y# Count the number of zeros
<-sum(new.y==0)
zeros.sim[i] }
We can then plot the distribution of \(T\) when the null hypothesis is true and overlay the number of zeros from our actual data set (Figure 17.1).
# T for our data
<-sum(slugs$slugs==0)) (T
[1] 34
hist(zeros.sim, xlab="# of zeros", ylab="Density", main="", col="gray", xlim=c(0, 35))
abline(v = T, col = "blue")
Lastly, we calculate the p-value as the probability of getting a value of \(T \ge 34\) when the null hypothesis is true:
#pvalue
sum(zeros.sim>=T)/nsims
[1] 0
Clearly, there are more zeros in the observed data than one might expect if the data were Poisson distributed with separate \(\lambda\)s for each site.
17.4 Zeros and the Negative Binomial model
Before considering zero-inflated models, it is important to recognize that the Negative Binomial distribution often provides a reasonable fit to data with many zeros. This has been my experience, and is also the view that has been expressed by Paul Allison, a statistician with lots of experience analyzing real data - see his blog here.
For example, if we repeat the goodness-of-fit test from the last section with a negative binomial model, we find that we do not reject the null hypothesis (Figure 17.2).
# Fitted model
<-MASS::glm.nb(slugs~field, data=slugs)
fit.nb
# Number of simulations
<-10000
nsims
# Set up matrix to hold goodness-of-fit statistics
<-matrix(NA, nsims, 1)
zeros.sim<-nrow(slugs) # number of observations
nobs
# Extract the estimated coefficients and their asymptotic variance/covariance matrix
# Use these values to generate nsims new beta's
<-MASS::mvrnorm(nsims, coef(fit.nb), vcov(fit.nb))
beta.hat<-rnorm(nsims, fit.nb$theta, fit.nb$SE.theta)
theta.hat
# Design matrix for creating new lambda^'s
<-model.matrix(fit.nb)
xmatfor(i in 1:nsims){
# Generate lambda^'s and thetas
<-exp(xmat%*%beta.hat[i,])
mus
# Generate new simulated values
<-rnbinom(nobs, mu = mus, size=theta.hat[i])
new.y
# Count the number of zeros
<-sum(new.y==0)
zeros.sim[i] }
Warning in rnbinom(nobs, mu = mus, size = theta.hat[i]): NAs produced
# T for our data
hist(zeros.sim, xlab="# of zeros", ylab="Density", main="", col="gray")
abline(v = T, col = "blue")
#pvalue
<-is.na(zeros.sim)!=TRUE # some NAs due to getting negative thetas
indsum(zeros.sim[ind]>=T)/sum(ind)
[1] 0.4681468
17.5 Hurdle models
There are two general classes for modeling zero-inflated data, Hurdle models and zero-inflated mixture models. We will briefly introduce hurdle models here but then focus on the latter class of model for the rest of this section. Hurdle models contain two different sub-components:
- A model for whether or not an observation is 0.
- A truncated count model for the non-zero observations.
These models can be fit using a two-step approach:
- Create a data set with new response variable, \(Z_i\), indicating whether the original response is a 0 or not:
\[Z_{i} = \left\{ \begin{array}{ll} 1 & \mbox{if } Y_i = 0\\ 0 & \mbox{if } Y_i \ne 0 \end{array} \right.\]
Model \(Z_i\) using logistic regression:
\[ Z_i \sim Bernoulli(p_i) \\ \] \[ logit(p_i) = \beta_0 + \beta_1X_{i,1}+\ldots \beta_pX_{i,p} \tag{17.1}\]
- Create a second data set containing only the non-zero observations, \(\widetilde{Y}_i\). Model \(\widetilde{Y}_i\) using a truncated count distribution, usually either a truncated-Poisson or truncated Negative Binomial. A truncated distribution is one in which we modify the probability mass function to exclude values below a threshold (in this case we exclude the possibility of a zero; see Section 17.5.1).
Alternatively, hurdle models can be fit in one step using the zeroinfl
function in the pscl
library. Fitting the model in a single step can be accomplished using maximum likelihood with the following probability mass function:
\[P(Y=y) = f(y; \pi, \lambda) =\left\{\begin{array}{ll} \pi & \mbox{if } y = 0\\ (1-\pi)f_T(y; \lambda) & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\]
where \(f_T(y; \lambda)\) is a truncated count distribution.
17.5.1 Truncated distributions
Remember our rule for calculating conditional probabilities: P(A|B)=P(A and B)/P(B) (Chapter 9). We can use this rule to determine the probability mass function for a truncated count distribution (truncated at 0):
\(f_T(y; \lambda)=P(Y = y | Y > 0) = \frac{P(Y=y)}{P(Y>0)}= \frac{f(y; \lambda)}{(1-f(0; \lambda))}\)
where \(f(y; \lambda)\) is the probability mass function for the non-truncated distribution and \(f(0; \lambda) = P(Y=0)\).
For example, the probability mass function for a truncated Poisson distribution with parameter \(\lambda\) would be given by:
\[P(Y=y | y > 0) = \frac{\frac{e^{-\lambda}\lambda^{y}}{y!}}{1-e^{-\lambda}}= \frac{e^{-\lambda}\lambda^{y}}{y!(1-e^{-\lambda})}\]
We could write a function to calculate these probabilities as below. We compare non-truncated and truncated Poisson distributions with the same \(\lambda\) in Figure 17.3.
par(mfrow=c(1,2))
<-function(x, lambda){
dtpois0 + I(x>0)*dpois(x, lambda)/(1-dpois(0, lambda))
}
It is also straightforward to allow the log mean of the non-zero observations to be modeled as a linear function of predictor variables:
\[ Y_i | Y_i > 0 \sim Poisson(\lambda_i) \\ \] \[\log(\lambda_i) = \beta_0 + \beta_1X_{1,i} +\ldots \beta_pX_{p,i} \tag{17.2}\]
A few important notes:
The same (or different) variables may be included in the two model sub-components – i.e., the model for whether or not an observation is 0 (logistic component of the model; Equation 17.1) and the model for the mean of the non-zero observations (Equation 17.2).
\(\lambda_i\) describes the mean of the non-zero observations, not the overall mean of the observations, which is given by: \(E[Y_i|X_i] = (1-p_i)\frac{\lambda_i}{1-e^{-\lambda_i}}\).
For more detail, along with expressions for \(E[Y_i|X_i]\) and \(Var[Y_i|X_i]\) for Poisson and Negative Binomial hurdle models, see page 288 of Zuur et al. (2009).
17.5.2 Hurdle models for continuous response data
The same general approach can be used to model continuous response data with many zeros. For continuous data, the non-zero observations can be modeled using log-normal or gamma distributions. Since these distributions already exclude zeros, there is no need to derive truncated versions of them. Alternatively, it is possible to use a truncated version of continuous probability distributions that allow for negative responses. For example, a truncated Normal distribution could be constructed using \(P(Y | Y>0) = \frac{f(y; \mu, \sigma^2)}{1-F(0; \mu, \sigma^2)}\) where \(F(y) = P(Y \le y)\). We could create a function for calculating the probability density function of a truncated Normal as follows:
<-function(x, mean, sd){
dtnorm0+I(x>0)*dnorm(x, mean, sd)/(1-pnorm(0, mean, sd))
}
17.6 Zero-inflated mixture models
In the next section, we will model counts of the number of fish that were caught by visitors of a state park. Not everyone that provided data actually fished (and, there is no way of knowing who went fishing and who didn’t since the park did not ask this question of survey respondents). It is easy to imagine two different ways to end up with 0 fish at the end of a camping trip: either you didn’t fish at all or you fished but were not successful. Some predictors may be uniquely suited for modeling whether or not you fished, while others may predict both whether you fished and also how successful you were. Zero-inflated mixture models provide a formal way to model these separate processes.
Like hurdle models, zero-inflated mixture models also have two model sub-components, both of which allow for the possibility of a 0:
- A model for whether or not an observation is an inflated 0.
- A count model that also allows for the possibility of a 0.
17.6.1 Marginal distribution
Let’s begin by considering a zero-inflated Poisson model without any covariates. Let \(Y_i\) be our zero-inflated response variable. We can think of \(Y_i\) as being determined by two random processes. First, a coin is flipped to determine if \(Y_i\) is an inflated zero; we get an inflated zero with probability \(\pi\). If \(Y_i\) is not an inflated zero, then we generate a Poisson random number. This allows us to get a 0 in two ways:
- we get an inflated zero with probability \(\pi\)
- we get a non-inflated zero by: 1) not having an inflated zero (occurs with probability \(1- \pi)\), and 2) the Poisson random variable is 0 (occurs with probability \(f(0) =\exp(-\lambda_i)\)). Combining these two probabilities using P(A and B) = P(A)P(B|A), we get the probability of a obtaining a 0 that is not inflated, which is equal to \((1-\pi)e^{-\lambda}\).
To get any non-zero number, we have to: 1) not have an inflated zero (occurs with probability \(1-\pi\)), and 2) the Poisson random variable has to take on the specific value (this occurs with probability \(f(y; \lambda)\)). Thus, the probability that we get a non-zero response, \(y\), is equal to \((1-\pi)f(y; \lambda)\).
Putting all of this together gives us the probability mass function for the zero-inflated Poisson model:
\(P(Y=y) = f(y; \pi, \lambda) =\left\{\begin{array}{ll} \pi + (1-\pi)e^{-\lambda} & \mbox{if } y = 0\\ (1-\pi)\frac{e^{-\lambda}\lambda^y}{y!} & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
A zero-inflated version of the Negative Binomial can be constructed in much the same way, noting that the probability Mass Function for the Negative Binomial distribution is given by: \(f(y; \mu, \theta) = {y+\theta-1 \choose y}\left(\frac{\theta}{\mu+\theta}\right)^{\theta}\left(\frac{\mu}{\mu+\theta}\right)^y\) and \(f(0; \mu, \theta) = \left(\frac{\theta}{\mu+\theta}\right)^\theta\).
Thus:
\(f(y; \mu, \theta) =\left\{ \begin{array}{ll} \pi + (1-\pi)\left(\frac{\theta}{\mu+\theta}\right)^\theta & \mbox{if } y = 0\\ (1-\pi){y+\theta-1 \choose y}\left(\frac{\theta}{\mu+\theta}\right)^{\theta}\left(\frac{\mu}{\mu+\theta}\right)^y & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
17.6.2 Zero-inflated mixture models defined using a latent variable
In my description of the zero-inflated mixture model in the previous section, I alluded to two random processes, the first of which was a coin flip to determine whether the response was an inflated 0. The second random process generated a Poisson random number (but only when the first process did not result in an inflated zero). It is often useful to formalize this idea by using a latent or unobserved variable, \(Z_i\), to define the zero-inflation process. Specifically, we may define our zero-inflated Poisson model as:
\[Z_i \sim Bernouli(p_i)\] \[Y_i | Z_i = 0 \sim Poisson(\lambda_i)\]
Alternatively, we may write:
\[Z_i \sim Bernouli(p_i)\] \[Y_i | Z_i \sim Poisson((1-Z_i)\lambda_i)\]
We will use the latter approach to define zero-inflated models in JAGS. In my opinion, this is an area where Bayesian methods shine – they provide an easy way to fit models that can be formulated in terms of latent (unobserved) variables.
As with hurdle models, it is important to understand that \(\lambda_i\) (or \(\mu_i\) if fitting a zero-inflated Negative Binomial) describes the mean, conditional on the observation not being an inflated zero (\(E[Y_i|X_i, Z_i = 0]\)). The unconditional mean and variance are given by1:
- \(E[Y_i|X_i] = (1-p_i)\lambda_i\) and \(E[Y_i|X_i] = (1-p_i)\mu_i\) for the Poisson and Negative Binomial distributions, respectively.
- \(Var[Y_i|X_i] = (1-p_i)(\lambda_i+p_i\lambda_i^2)\) and \(Var[Y_i|X_i] = (1-p_i)(\mu_i+\frac{\mu_i^2}{\theta})+\mu_i^2p_i(1-p_i)\) for the Poisson and Negative Binomial distributions, respectively.
These expressions are needed when constructing Pearson residuals. As with hurdle models, it is also easy to incorporate predictors that influence the probability of getting an inflated-zero or the mean of our non-inflated observations. Let’s take a look with an applied example.
17.7 Example: Fishing success in state parks
Here, we will consider the zeroinfl
function in the pscl
package for fitting zero-inflated versions of the Poisson and Negative Binomial models (Jackman 2020). The zeroinfl
function can also be used to fit hurdle models. We will consider a data set and example from UCLA’s Statistical Consulting website. The data have also been incorporated into the Data4Ecologists
package and can be accessed using:
data(fish)
The data set is constructed from surveys of 250 groups that visited a state park and contains the following variables:
count
= reported number of fish the group caughtchild
= how many children were in the grouppersons
= the number of people that were in the groupcamper
= 1 if the group brought a camper to the campground and 0 otherwise
On the UCLA page, they considered the following zero-inflated model:
\[\begin{gather} Z_i | X_i \sim Bernoulli(p_i)\\ \text{logit}(p_i) = \gamma_0 + \gamma_1persons_i\\ Y_i|X_i, Z_i \sim Poisson((1-Z_i)\lambda_i)\\ \log(\lambda_i) = \beta_0 + \beta_1 child_i+\beta_2 camper_i \end{gather}\]
I.e., they assumed that the probability of an inflated zero depended on the number of people in the group and that the mean number of fish caught, given the observation was not an inflated zero, depended on the number of children in the group and whether the group brought a camper.
As someone who likes to fish and also camp with family members (3 of whom were once young children), I think a better model would be:
\[\begin{gather} Z_i | X_i \sim Bernoulli(p_i) \\ logit(p_i) = \gamma_0 + \gamma_1child_i \\ Y_i|X_i, Z_i \sim Poisson((1-Z_i)\lambda_i) \\ \log(\lambda_i) = \beta_0 + \beta_1 child_i+\beta_2 camper_i + \beta_3 person_i \end{gather}\]
We can fit this model as follows:
<- zeroinfl(count ~ child + camper +persons | child, data = fish, dist="poisson")
m.zip summary(m.zip)
Call:
zeroinfl(formula = count ~ child + camper + persons | child, data = fish,
dist = "poisson")
Pearson residuals:
Min 1Q Median 3Q Max
-1.4052 -0.7178 -0.4505 -0.1199 26.9596
Count model coefficients (poisson with log link):
Estimate Std. Error z value Pr(>|z|)
(Intercept) -1.05721 0.18123 -5.834 5.42e-09 ***
child -1.16755 0.09471 -12.327 < 2e-16 ***
camper 0.77091 0.09384 8.215 < 2e-16 ***
persons 0.88856 0.04663 19.056 < 2e-16 ***
Zero-inflation model coefficients (binomial with logit link):
Estimate Std. Error z value Pr(>|z|)
(Intercept) -0.9150 0.2503 -3.655 0.000257 ***
child 1.1857 0.2654 4.468 7.89e-06 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Number of iterations in BFGS optimization: 12
Log-likelihood: -766 on 6 Df
The variables on the right hand side of the “|” specify the variables that influence the probability of an inflated zero (in this case, child
). We can interpret the coefficients of the zero-inflation part of the model in the same way as we did for logistic regression. We see that the odds of an inflated zero increases by a factor of exp(1.1857) = 3.27 for each additional child in the group. Furthermore, a group with 3 children is exp(1.1857*3) = 35 times more likely to be an inflated zero than a group with 0 children. Lastly, the probability that a group with 3 children is an inflated zero is given by \(\frac{\exp(-0.9150+3 \times 1.1858)}{1+\exp(-0.9150+3 \times 1.1858)}\) and can be calculated using the plogis
function:
plogis(-0.9150+3*1.1857)
[1] 0.9335224
I am not too surprised – it is a lot of work to take 3 children fishing (even if it is incredibly rewarding)! Looking at the count part of the model, we see that groups with children catch fewer fish, whereas larger groups and groups that have a camper tend to catch more fish. Perhaps those with a camper also have better equipment (depth finders, a nice boat, etc). As with standard count models, we can estimate multiplicative effects on the mean number of fish caught, given the observation is not an inflated zero, by exponentiating these parameters. For example, for non-inflated observations, groups with a camper are expected to catch exp(0.77091) = 2.16 times as many fish as groups without a camper.
We can also fit a zero-inflated negative binomial model by changing the distribution argument (dist
):
\[\begin{gather} Z_i |X_i \sim Bernoulli(p_i) \\ logit(p_i) = \gamma_0 + \gamma_1child_i \\ Y_i|X_i, Z_i \sim Negative Binomial((1-Z_i)\mu_i, \theta) \\ \log(\mu_i) = \beta_0 + \beta_1 child_i+\beta_2 camper_i + \beta_3 person_i \end{gather}\]
<- zeroinfl(count ~ child + camper +persons | child,
m.zinb data = fish, dist = "negbin")
summary(m.zinb)
Call:
zeroinfl(formula = count ~ child + camper + persons | child, data = fish,
dist = "negbin")
Pearson residuals:
Min 1Q Median 3Q Max
-0.71608 -0.54838 -0.34328 -0.02089 15.00817
Count model coefficients (negbin with log link):
Estimate Std. Error z value Pr(>|z|)
(Intercept) -1.6600 0.3197 -5.192 2.08e-07 ***
child -1.2056 0.2715 -4.441 8.95e-06 ***
camper 0.5834 0.2379 2.452 0.0142 *
persons 1.0516 0.1110 9.477 < 2e-16 ***
Log(theta) -0.5824 0.1823 -3.195 0.0014 **
Zero-inflation model coefficients (binomial with logit link):
Estimate Std. Error z value Pr(>|z|)
(Intercept) -4.4312 1.5168 -2.921 0.003484 **
child 2.9268 0.8481 3.451 0.000559 ***
---
Signif. codes: 0 '***' 0.001 '**' 0.01 '*' 0.05 '.' 0.1 ' ' 1
Theta = 0.5585
Number of iterations in BFGS optimization: 17
Log-likelihood: -399.9 on 7 Df
Note that the zeroinfl
function parameterizes the overdispersion parameter, \(\theta\) on the log-scale. This ensures that the estimate of \(\theta\), is positive (exp(Log(theta)
) = exp(-0.5823) = 0.5586, which is also printed at the bottom of the summary output). The qualitative conclusions are similar to those of the zero-inflated Poisson model, but the coefficients and p-values have changed slightly.
We can compare the two zero-inflated models and also poisson and negative binomial models using AIC (below):
<- glm(count ~ child + camper + persons, family = poisson, data = fish)
m.pois<- glm.nb(count ~ child + camper + persons, data = fish)
m.nb AIC(m.pois, m.nb, m.zip, m.zinb)
df AIC
m.pois 4 1682.1450
m.nb 5 820.4440
m.zip 6 1544.0557
m.zinb 7 813.8197
This comparison suggests that the zero-inflated negative binomial model provides the best fit to the data, with the non-zero inflated negative binomial model not far behind. The Poisson and zero-inflated Poisson models, by comparison, provide poor fits to the data. My experience, and that of others, is that a Negative Binomial model (without zero-inflation) often “wins” followed by the zero-inflated negative binomial distribution (see e.g., Gray 2005; Warton 2005; Sileshi 2008 for comparative studies).
Lastly, we exponentiate the parameters in our models to summarize effect sizes using incidence ratios for our count models (Table 17.1) and incidence and odds ratios for our zero-inflated models (Table 17.2).
Poisson | Negative Binomial | |||
Predictors | Incidence Rate Ratios | CI | Incidence Rate Ratios | CI |
(Intercept) | 0.14 | 0.10 – 0.18 | 0.20 | 0.10 – 0.38 |
child | 0.18 | 0.16 – 0.22 | 0.17 | 0.11 – 0.24 |
camper | 2.54 | 2.14 – 3.03 | 1.86 | 1.17 – 2.95 |
persons | 2.98 | 2.76 – 3.22 | 2.89 | 2.31 – 3.66 |
AIC | 1682.145 | 820.444 |
Zero-Inflated Poisson | Zero-Inflated Negative Binomial | |||
Predictors | Incidence Rate Ratios | CI | Incidence Rate Ratios | CI |
(Intercept) | 0.35 | 0.24 – 0.50 | 0.19 | 0.10 – 0.36 |
child | 0.31 | 0.26 – 0.37 | 0.30 | 0.18 – 0.51 |
camper | 2.16 | 1.80 – 2.60 | 1.79 | 1.12 – 2.86 |
persons | 2.43 | 2.22 – 2.66 | 2.86 | 2.30 – 3.56 |
Zero-Inflated Model | ||||
(Intercept) | 0.40 | 0.25 – 0.65 | 0.01 | 0.00 – 0.23 |
child | 3.27 | 1.95 – 5.51 | 18.67 | 3.54 – 98.41 |
AIC | 1544.056 | 813.820 |
17.8 Goodness-of-fit
The AIC comparisons only tell us which of our models is “best”, but they do not tell us if any of the models are actually any good. We could modify our goodness-of-fit test based on Pearson residuals. This would require generating Bernoulli random variables for the zero-inflation process and either Poisson or Negative Binomial random variables for the non-inflated zero observations. We demonstrate this approach when fitting the zero-inflated negative binomial model in JAGS (see Section 17.9). We can also explore plots of the observed counts or Pearson residuals versus fitted values (Figure 17.4).
$predict.m.zinb<-predict(m.zinb, type="response")
fish$presiduals.m.zinb<-resid(m.zinb, type="pearson")
fish<-ggplot(fish, aes(predict.m.zinb, count))+geom_point()+
p1geom_jitter()+geom_abline(intercept=0,slope=1)+
xlab("Predicted")+ylab("Observed")
<-ggplot(fish, aes(predict.m.zinb, presiduals.m.zinb)) +
p2geom_point() +
geom_jitter()+
geom_abline(intercept=0,slope=0) +
geom_smooth() +
xlab("Predicted")+ylab("Pearson Residual")
+p2 p1
`geom_smooth()` using method = 'loess' and formula = 'y ~ x'
Inspecting these on these two plots (Figure 17.4), there are 2 observations we might want to look into further:
- the group that reported catching ~150 fish (is that even possible?)
- the observation with the large Pearson residual (\(\sim 15\)).
We can use the which.max
function to identify these observations:
<-which.max(fish$count) # Max count
ind1<-which.max(fish$presiduals.m.zinb) # Max Pearson residual
ind2c(ind1, ind2),] fish[
count livebait camper persons child predict.m.zinb presiduals.m.zinb
89 149 1 1 4 0 22.605695 4.091124
138 31 1 0 3 1 1.092768 15.008168
The group that caught 149 fish had 4 people with no children. The observation with the large Pearson residual was for a group of 3 people, one of which was a child, that caught 31 fish. Based on the Pearson residual, we might conclude that the latter observation is actually more of an outlier!
17.9 Bayesian implementation
Here, we follow the approach taken by Kéry (2010). Note that Kery parameterizes the model in terms of \(\psi = 1- \pi\) = the probability of a non zero-inflated response. In comparison to the formulation using zerolinfl
, we have:
Parameterization used by zeroinfl
:
\(P(Y=y) = f(y) =\left\{\begin{array}{ll} \pi + (1-\pi)e^{-\lambda} & \mbox{if } y = 0\\ (1-\pi)\frac{e^{-\lambda}\lambda^y}{y!} & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
ZIP model (Kery):
\(P(Y=y) = f(y) =\left\{\begin{array}{ll} 1-\psi + \psi e^{-\lambda} & \mbox{if } y = 0\\ \psi\frac{e^{-\lambda}\lambda^y}{y!} & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
And, for the Negative Binomial model:
zeroninfl
parameterization:
\(f(y) =\left\{ \begin{array}{ll} \pi + (1-\pi)\left(\frac{\theta}{\mu+\theta}\right)^\theta & \mbox{if } y = 0\\ (1-\pi){y+\theta-1 \choose y}\left(\frac{\theta}{\mu+\theta}\right)^{\theta}\left(\frac{\mu}{\mu+\theta}\right)^y & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
ZINB model (Kery):
\(f(y) =\left\{ \begin{array}{ll} 1-\pi + \pi\left(\frac{\theta}{\mu+\theta}\right)^\theta & \mbox{if } y = 0\\ \pi{y+\theta-1 \choose y}\left(\frac{\theta}{\mu+\theta}\right)^{\theta}\left(\frac{\mu}{\mu+\theta}\right)^y & \mbox{if } y = 1, 2, 3, \ldots \end{array} \right.\)
The only difference is that the coefficients in the logistic part of the model will be of the opposite sign since we are now modeling the probability of a non-inflated zero rather than the probability of an inflated zero.
We use the latent variable approach to implement the Bayesian zero-inflated Negative Binomial model. Specifically, we will create a (partially) latent variable, I.fish
(rather than \(Z_i\)), to indicate whether an observation is NOT an inflated 0. We will think of I.fish
as an indicator of whether or not the group went fishing (if so, we do not have an inflated 0). We then model count
as a Negative Binomial random variable with mean = I.fish
\(\times\mu_i\):
\[I.fish_i \sim Bernouli(p_i)\] \[logit(p_i) = \gamma_0 +\gamma_1 child_i\] \[count_i | I.fish_i \sim \mbox{Negative Binomial}(I.fish_i \times \mu_i, \theta)\] \[\log(\mu_i) = \beta_0 + \beta_1 child_i+\beta_2 camper_i + \beta_3 person_i\]
What makes this model different from others that we have considered so far is that we do not actually have a variable in our data set called I.fish
that takes on a value of 1 when the group went fishing and 0 otherwise. We will create the I.fish
variable, initially setting all of its values to NA (missing). We then set I.fish
= 1 for all groups that caught a fish (these cannot be inflated zeros), leaving the value of I.fish
= NA for all other observations. JAGS will impute values for I.fish
for the observations with missing values during each MCMC iteration using information from the predictor variables for those individuals.
<-rep(NA, nrow(fish))
I.fish$count>0]<-1 # these people caught fish and are NOT inflated zeros I.fish[fish
Think-Pair-Share: How many priors will we need for this model?
<-function(){
znb
# Priors for count model
for(i in 1:4){
~ dnorm(0,0.001)
beta.c[i]
}
# Priors for zero-inflation model (fit on a logit scale)
for(i in 1:2){
~ dnorm(0,1/3)
beta.zi[i]
}
# Overdispersion parameter
~ dunif(0,50)
theta
# Likelihood
for(i in 1:n){
# zero-inflation part (logit prob NOT inflated 0, i.e., "went fishing"))
~ dbern(psi[i]) # Not zero-inflated (i.e., "went fishing")
I.fish[i] <- beta.zi[1] + beta.zi[2]*child[i]
logitpsi[i] <- exp(logitpsi[i])/(1+exp(logitpsi[i])) # prob Not inflated 0
psi[i]
# Count part
~ dnegbin(p[i], theta)
count[i] <- beta.c[1] + beta.c[2]*child[i] + beta.c[3]*camper[i] + beta.c[4]*persons[i]
log.mu[i] <- exp(log.mu[i])
mu[i] <- mu[i]*I.fish[i]
mu.eff[i] <- theta/(theta+mu.eff[i])
p[i]
# Mean and variances of the observations
# Derived using: Var(Count) = E[var(count|z)] + Var[E(count|z)]
<- mu[i]*psi[i]
Ey[i] <- psi[i]*(mu[i] + mu[i]^2/theta) + mu[i]^2*(psi[i]*(1 - psi[i]))
Vary[i]
# Generate "new" data
~ dbin(psi[i],1)
I.fish.new[i] <- mu[i]*I.fish.new[i]
mu.eff.new[i] <- theta/(theta + mu.eff.new[i])
p.new[i] ~ dnegbin(p.new[i], theta)
count.new[i]
# Pearson residuals
<- (count[i] - Ey[i])/sqrt(Vary[i]) # Pearson Resid
presi[i] <- (count.new[i] - Ey[i])/sqrt(Vary[i])
presi.new[i]
# Discrepancy measures
<- pow(presi[i], 2)
D[i] <- pow(presi.new[i],2)
D.new[i]
}<- sum(D[])
fit <- sum(D.new[])
fit.new
}
# Bundle data
<- list(count=fish$count,
jagsdata child=fish$child,
camper = as.numeric(fish$camper) -1,
persons = fish$persons,
n = nrow(fish),
I.fish = I.fish )
# Parameters to estimate
<- c("beta.zi", "beta.c", "Ey", "psi", "mu", "presi",
params "presi.new", "fit", "fit.new", "theta", "I.fish")
# Start sampler
<- jags.parallel(data = jagsdata, parameters.to.save = params,
out.znb model.file = znb, n.thin = 10, n.chains = 3, n.burnin = 4000,
n.iter= 15000)
MCMCsummary(out.znb, params = c("beta.zi", "beta.c", "theta"))
mean sd 2.5% 50% 97.5% Rhat n.eff
beta.zi[1] 3.2260341 0.7466206 1.9449418 3.1598389 4.8895102 1.00 901
beta.zi[2] -2.0797874 0.6857655 -3.2646778 -2.1018733 -0.7394781 1.03 469
beta.c[1] -1.0069410 0.3067349 -1.6079080 -1.0078509 -0.4026453 1.00 1467
beta.c[2] -1.2214089 0.2898108 -1.8184555 -1.2153159 -0.6678798 1.01 1168
beta.c[3] 0.5792946 0.2369349 0.1058049 0.5815875 1.0401012 1.00 3007
beta.c[4] 1.0452877 0.1106891 0.8328963 1.0447593 1.2676449 1.00 1514
theta 0.5986998 0.1081040 0.4184918 0.5881724 0.8413517 1.01 2073
As always, we should explore traceplots and density plots to make sure that our sampler has converged (Figure 17.5, Figure 17.6).
denplot(out.znb, parms=c("beta.zi", "beta.c", "theta"), ask=FALSE)
traplot(out.znb, parms=c("beta.zi", "beta.c", "theta"), ask=FALSE)
We can also calculate the Bayesian p-value for the goodness-of-fit test. Although the p-value is fairly small, the conclusions depend critically on the one outlier noted previously with Pearson residual ~ 15. If we drop that observation, the p-value increases substantially (results not shown).
<- MCMCpstr(out.znb, params = c("fit", "fit.new"), type = "chains")
fitstats <- fitstats$fit.new >= fitstats$fit
T.extreme <- mean(T.extreme)) (p.val
[1] 0.05787879
Lastly, it may be interesting to look at the imputed values for I.Fish
and how they relate to the number of children in each group. Below, we plot the distribution of posterior means for I.fish
at each level of child
.
$I.fish.hat<-out.znb$BUGSoutput$mean$I.fish
fishggplot(fish, aes(x = as.factor(child), y = I.fish.hat)) +
geom_boxplot() +
xlab("Number of Children")+ylab("Went fishing?") +
geom_jitter(color = "black", size = 1, alpha = 0.9)
I.fish
, an indicator that the observation is not an inflated zero, as a function of child
.
We see that I.fish
is rarely = 1 in groups with 2 or 3 children. This makes sense in light of the fact that no group with 3 children actually caught any fish and the mean number of fish caught for groups with 2 children was less than 1 fish!
%>% group_by(child) %>% dplyr::summarize(meanfish=mean(count)) fish
# A tibble: 4 × 2
child meanfish
<int> <dbl>
1 0 5.19
2 1 1.76
3 2 0.212
4 3 0
17.10 Implementation using glmmTMB
Although the zeroinfl
function in the pscl
package is probably the most widely used function for fitting zero-inflation models, it is also possible to fit zero-inflated Poisson and zero-inflated negative binomial models using the glmmTMB
function in the glmmTMB
package (Brooks et al. 2017). The main advantage to using glmmTMB
is that it makes it possible to include random effects (e.g., when dealing with non-independent data; see Chapter 18). The glmmTMB
function has a separate argument, ziformula
, for specifying the zero-inflation part of the model. It also uses the argument family
to specify the distribution (either poisson
or nbinom2
). Below, we demonstrate the equivalence of zero-inflated Poisson models fit using glmmTMB
and zeroinfl
.
# Equivalent zero-inflated Poisson models
library(glmmTMB)
<- glmmTMB(count ~ child + camper + persons, ziformula = ~ child,
m.zip.TMB data = fish, family = poisson)
m.zip
Call:
zeroinfl(formula = count ~ child + camper + persons | child, data = fish,
dist = "poisson")
Count model coefficients (poisson with log link):
(Intercept) child camper persons
-1.0572 -1.1675 0.7709 0.8886
Zero-inflation model coefficients (binomial with logit link):
(Intercept) child
-0.915 1.186
m.zip.TMB
Formula: count ~ child + camper + persons
Zero inflation: ~child
Data: fish
AIC BIC logLik df.resid
1544.0557 1565.1845 -766.0279 244
Number of obs: 250
Fixed Effects:
Conditional model:
(Intercept) child camper persons
-1.0572 -1.1675 0.7709 0.8886
Zero-inflation model:
(Intercept) child
-0.915 1.186
And, here, the equivalence between zero-inflated Negative Binomial models fit using glmmTMB
and zeroinfl
.
# Equivalent zero-inflated Negative Binomial Models
<- glmmTMB(count ~ child + camper + persons, ziformula = ~ child,
m.zinb.TMB data = fish, family = nbinom2)
m.zinb
Call:
zeroinfl(formula = count ~ child + camper + persons | child, data = fish,
dist = "negbin")
Count model coefficients (negbin with log link):
(Intercept) child camper persons
-1.6600 -1.2056 0.5834 1.0516
Theta = 0.5585
Zero-inflation model coefficients (binomial with logit link):
(Intercept) child
-4.431 2.927
m.zinb.TMB
Formula: count ~ child + camper + persons
Zero inflation: ~child
Data: fish
AIC BIC logLik df.resid
813.8197 838.4700 -399.9099 243
Number of obs: 250
Dispersion parameter for nbinom2 family (): 0.559
Fixed Effects:
Conditional model:
(Intercept) child camper persons
-1.6600 -1.2056 0.5834 1.0516
Zero-inflation model:
(Intercept) child
-4.430 2.926
17.11 References
These expressions can be derived fairly easily using \(E[Y]=E_X(E[Y|X])\) and \(Var[Y]=Var_X(E[Y|X]) + E_X(Var[Y|X])\).↩︎