Setup

library(ggplot2)
set.seed(2)


Model, Data, & Likelihood

Let’s build a simple MCMC sampler using the Metropolis algorithm. First, we’ll set up our model. We’ll suppose the data (\(z\)) comes from a Poisson distribution, with an unknown rate \(\lambda\): \[ z \sim \mathcal{P}(\lambda) \] Then our the likelihood for a single data point \(z_i\) is given by: \[ P(z_i \,|\, \lambda) = f(z_i \,|\, \lambda) = \frac{\lambda^{z_i} e^{-\lambda}}{z_i!}\] and our overall likelihood for a full data set \(z = \{z_1,\dots, z_n\}\) is given by: \[ P(z \,|\, \lambda) = \prod_{i=1}^n f(z_i \,|\, \lambda) \]

likelihood = function(data,lambda){prod(dpois(data,lambda))}


From the above model, let’s simulate a little data set that we can use for parameter estimation:

data = rpois(20,2)


Prior Distribution

Now, we need a prior for our unknown parameter \(\mu\). We’ll choose a normal distribution for the prior of \(\mu\). But you can also try other priors (e.g. uniform) too! For our prior \(P(\mu)\), we’ll choose \(\mu \sim \mathcal{N}(1,1.5)\).

# Set up prior
prior = function(lambda){dnorm(lambda,1,1.5)}

# Plot the prior
plot(seq(0,5,0.1),prior(seq(0,5,0.1)),type='l',xlab="λ", ylab="density")


Setting up the Metropolis algorithm for MCMC

As we saw in class, the Metropolis algorithm for MCMC is:

  1. Initialization
    • Choose a starting point in parameter space, say \(\lambda = 5\).
    • Decide on a proposal distribution to choose the next point to sample—this distribution is often somewhat arbitrary, but different choices will make it easier or harder to converge to the stationary distribution. In our case we’ll choose a normal distribution with mean given by the current sampled parameter value (call this \(\lambda_{curr}\)), and standard deviation of 0.5, \(\mathcal{N}(\lambda_{curr}, 0.5)\).
    • Decide how many samples you want to take—in our case, let’s take 5000.

We’ll also make a variable, samples, to hold our MCMC samples as we go.

lambda.curr = 5 #start value for lambda
proposaldist = function(paramval){rnorm(1,paramval,0.5)}
numsteps = 5000
samples = numeric(numsteps)
  1. For each iteration,
    • Use the proposal distribution to draw a new proposed sample, \(\lambda_{next}\)

    • Calculate the product of the likelihood and the prior, \(P(z \,|\, \lambda) \cdot P(\lambda)\), for both the current and newly proposed values of \(\lambda\). Note that \(P(z \,|\, \lambda) \cdot P(\lambda)\) is proportional to the posterior (it is just missing the denominator from Bayes’ theorem), so our sampler will recover the posterior distribution.

    • Calculate the acceptance ratio, \(a = \frac{P(z \,|\, \lambda_{next}) \cdot P(\lambda_{next})}{P(z \,|\, \lambda_{curr}) \cdot P(\lambda_{curr})}\).

    • If \(a\geq 1\), the new point is as good or better than the last, and we automatically accept.

    • If \(a<1\), then the new point \(\lambda_{next}\) is worse. In this case, we accept the new point with probability \(a\). If it is accepted, we set lambda.curr = lambda.next. If it is not accepted, we leave lambda.curr as is and re-draw lambda.next in the next iteration.

    • Also, don’t forget to record your lambda.curr samples at each iteration!

Code the algorithm above and generate 5000 samples for \(\lambda\). Here’s more explicit pseudocode for 2) to help you set things up:


Plotting the Results

Next, plot the chain—do you see burn-in? Roughly when does it converge to the stationary distribution?

plot(1:numsteps,samples, xlab='iteration',ylab='λ')

Lastly, if we plot the prior, likelihood, and posterior, we can see how the likelihood alters the prior to generate the posterior. We can also compare the sampled posterior matches the analytical posterior.

# take just the later part of the chain, after burn-in
posteriorsample = samples[500:numsteps]

# set up the prior & likelihood for line-plotting 
# (also re-scale them so they're on similar y-axis scales)

paramvalslist = seq(0,4,0.1) # parameter values list

#corresponding prior values
priorlist = prior(paramvalslist)
priorlist = priorlist/max(priorlist)

#likelihood values
likelihoodlist = numeric(length(paramvalslist))
for(i in 1:length(paramvalslist)){likelihoodlist[i] = likelihood(data,paramvalslist[i])}
likelihoodlist = likelihoodlist/max(likelihoodlist)

# This is the dataframe we'll use for ggplot
plotdata = data.frame(paramvalslist,priorlist,likelihoodlist)

ggplot(plotdata) + 
  geom_density(data=as.data.frame(posteriorsample),aes(posteriorsample, color='Sampled Posterior'),size=1) +
  geom_line(aes(paramvalslist,likelihoodlist, color='Likelihood'),size=1) + 
  geom_line(aes(paramvalslist,priorlist, color='Prior'),size=1) + 
  labs(x="µ", y="scaled density") + guides(color = guide_legend('')) + 
  scale_colour_brewer(palette = "Set1")


Extra problems