6.3 Rejection Sampling | Advanced Statistical Computing

Excerpt

The book covers material taught in the Johns Hopkins Biostatistics Advanced Statistical Computing course.


What do we do if we want to generate samples of a random variable with density f and there isn’t a built in function for doing this? If the random variable is of a reasonably low dimension (less than 10?), then rejection sampling is a plausible general approach.

The idea of rejection sampling is that although we cannot easily sample from f, there exists another density g, like a Normal distribution or perhaps a t-distribution, from which it is easy for us to sample (because there’s a built in function or someone else wrote a nice function). Then we can sample from g directly and then “reject” the samples in a strategic way to make the resulting “non-rejected” samples look like they came from f. The density g will be referred to as the “candidate density” and f will be the “target density”.

In order to use the rejections sampling algorithm, we must first ensure that the support of f is a subset of the support of g. If Xf is the support of f and Xg is the support of g, then we must have Xf⊂Xg. This makes sense: if there’s a region of the support of f that g can never touch, then that area will never get sampled. In addition, we must assume that c=supx∈Xff(x)g(x)<∞ and that we can calculate c. The easiest way to satisfy this assumption is to make sure that g has heavier tails than f. We cannot have that g decreases at a faster rate than f in the tails or else rejection sampling will not work.

The Algorithm

The rejection sampling algorithm for drawing a sample from the target density f is then

  1. Simulate U∼Unif(0,1).

  2. Simulate a candidate X∼g from the candidate density

  3. If U≤f(X)cg(X) then “accept” the candidate X. Otherwise, “reject” X and go back to the beginning.

The algorithm can be repeated until the desired number of samples from the target density f has been accepted.

As a simple example, suppose we wanted to generate samples from a N(0,1) density. We could use the t2 distribution as our candidate density as it has heavier tails than the Normal. Plotting those two densities, along with a sample from the t2 density gives us the picture below.

<span id="cb120-1"><span>set.seed</span>(<span>2017-12-4</span>)</span>
<span id="cb120-2"><span>curve</span>(<span>dnorm</span>(x), <span>-</span><span>6</span>, <span>6</span>, <span>xlab =</span> <span>"x"</span>, <span>ylab =</span> <span>"Density"</span>, <span>n =</span> <span>200</span>)</span>
<span id="cb120-3"><span>curve</span>(<span>dt</span>(x, <span>2</span>), <span>-</span><span>6</span>, <span>6</span>, <span>add =</span> <span>TRUE</span>, <span>col =</span> <span>4</span>, <span>n =</span> <span>200</span>)</span>
<span id="cb120-4"><span>legend</span>(<span>"topright"</span>, <span>c</span>(<span>"Normal density"</span>, <span>"t density"</span>), </span>
<span id="cb120-5">       <span>col =</span> <span>c</span>(<span>1</span>, <span>4</span>), <span>bty =</span> <span>"n"</span>, <span>lty =</span> <span>1</span>)</span>
<span id="cb120-6">x <span>&lt;-</span> <span>rt</span>(<span>200</span>, <span>2</span>)</span>
<span id="cb120-7"><span>rug</span>(x, <span>col =</span> <span>4</span>)</span>

Given what we know about the standard Normal density, most of the samples should be between −3 and +3, except perhaps in very large samples (this is a sample of size 200). From the picture, there are samples in the range of 4–6. In order to transform the t2 samples into N(0,1) samples, we will need to reject many of the samples out in the tail. On the other hand, there are two few samples in the range of [−2,2] and so we will have to disproportionaly accept samples in that range until it represents the proper N(0,1) density.

Before we move on, it’s worth noting that the rejection sampling method requires that we can evaluate the target density f. That is how we compute the rejection/acceptance ratio in Step 2. In most cases, this will not be a problem.

Properties of Rejection Sampling

One property of the rejection sampling algorithm is that the number of draws we need to take from the candidate density g before we accept a candidate is a geometric random variable with success probability 1/c. We can think of the decision to accept or reject a candidate as a sequence of iid coin flips that has a specific probability of coming up “heads” (i.e. being accepted). That probability is 1/c and we can calculate that as follows.

P(X accepted)=P(U≤f(X)cg(X))=∫P(U≤f(x)cg(x)|X=x)g(x)dx=∫f(x)cg(x)g(x)dx=1c

This property of rejection sampling has implications for how we choose the candidate density g. In theory, any density can be chosen as the candidate as long as its support includes the support of f. However, in practice we will want to choose g so that it matches f as closely as possible. As a rule of thumb, candidates g that match f closely will have smaller values of c and thus will accept candidates with higher probability. We want to avoid large values of c because large values of c lead to an algorithm that rejects a lot of candidates and has lower efficiency.

In the example above with the Normal distribution and the t2 distribution, the ratio f(x)/g(x) was maximized at x=1 (or x=−1) and so the value of c for that setup was 1.257, which implies an acceptance probability of about 0.8. Suppose however, that we wanted to simulate from a Uniform(0,1) density and we used an Exponential(1) as our candidate density. The plot of the two densities looks as follows.

<span id="cb121-1"><span>curve</span>(<span>dexp</span>(x, <span>1</span>), <span>0</span>, <span>1</span>, <span>col =</span> <span>4</span>, <span>ylab =</span> <span>"Density"</span>)</span>
<span id="cb121-2"><span>segments</span>(<span>0</span>, <span>1</span>, <span>1</span>, <span>1</span>)</span>
<span id="cb121-3"><span>legend</span>(<span>"bottomleft"</span>, <span>c</span>(<span>"f(x) Uniform"</span>, <span>"g(x) Exponential"</span>), <span>lty =</span> <span>1</span>, <span>col =</span> <span>c</span>(<span>1</span>, <span>4</span>), <span>bty =</span> <span>"n"</span>)</span>

Here, the ratio of f(x)/g(x) is maximized at x=1 and so the value of c is 2.718 which implies an acceptance probablity of about 0.37. While running the rejection sampling algorithm in this way to produce Uniform random variables will still work, it will be very inefficient.

We can now show that the distribution of the accepted values from the rejection sampling algorithm above follows the target density f. We can do this by calculating the distribution function of the accepted values and show that this is equal to F(t)=∫−∞tf(x)dx.

P(X≤t∣X accepted)=P(X≤t,X accepted)P(X accepted)=P(X≤t,X accepted)1/c=cEgE[1{x≤t}1{U≤f(x)cg(x)}|X=x]=cEg[1{X≤t}E[1{U≤f(x)cg(x)}|X=x]]=cEg[1{X≤t}f(X)cg(X)]=∫−∞∞1{x≤t}f(x)g(x)g(x)dx=∫−∞tf(x)dx=F(t) This shows that the distribution function of the candidate values, given that they are accepted, is equal to the distribution function corresponding to the target density.

A few further notes:

  1. We only need to know f and g up to a constant of proportionality. In many applications we will not know the normalizing constant for these densities, but we do not need them. That is, if f(x)=k1f⋆(x) and g(x)=k2g⋆(x), we can proceed with the algorithm using f⋆ and g⋆ even if we do not know the values of k1 and k2.

  2. Any number c′≥c will work in the rejection sampling algorithm, but the algorithm will be less efficient.

  3. Throughout the algorithm, operations can (and should!) be done on a log scale.

  4. The higher the dimension of f and g, the less efficient the rejection sampling algorithm will be.

  5. Whether c=∞ or not depends on the tail behavior of the the densities f and g. If g(x)↓0 faster than f(x)↓0 as x→∞, then f(x)/g(x)↑∞.

Empirical Supremum Rejection Sampling

What if we cannot calculate c=supx∈Xff(x)g(x) or are simply too lazy to do so? Fear not, because it turns out we almost never have to do so. A slight modification of the standard rejection sampling algorithm will allow us to estimate c while also sampling from the target density f. The tradeoff (there is always a tradeoff!) is that we must make a more stringent assumption about c, mainly that it is achievable. That is, there exists some value xc∈Xf such that f(xc)g(xc) is equal to supx∈Xff(x)g(x).

The modified algorithm is the empirical supremum rejection sampling algorithm of Caffo, Booth, and Davison. The algorithm goes as follows. First we must choose some starting value of c, call it c^, such that c^>1. Then,

  1. Draw U∼Unif(0,1).

  2. Draw X∼g, the candidate density.

  3. Accept X if U≤f(X)c^g(X), otherwise reject X.

  4. Let c^⋆=max{c^,f(X)g(X)}.

  5. Update c^=c^⋆.

  6. Goto Step 1.

From the algorithm we can see that at each iteration, we get more information about the ratio f(X)/g(X) and can update our estimate of c accordingly.

One way to think of this algorithm is to conceptualize a separate sequence Yi, which is 0 or 1 depending on whether Xi should be rejected (0) or accepted (1). This sequence Yi is the accept/reject determination sequence. Under the standard rejection sampling algorithm, the sequence Y~i is generated using the true value of c. Under the emprical supremum rejection sampling (ESUP) scheme, we generate a slightly different sequence Yi using our continuously updated value of c^.

If we drew values X1,X2,X3,X4,X5,X6,… from the candidate density g, then we could visualize the acceptance/rejection process as it might occur using the true value of c and our estimate c^.

Empirical supremum rejection sampling scheme.

Following the diagram above, we can see that using the estimate c^, there are two instances where we accept a value when we should have rejected it (X1 and X4). In every other instance in the sequence, the value of Yi was equal to Yi. The theory behind the ESUP algorithm is that eventually, the sequence Yi becomes identical to the sequence Yi and therefore we will accept/reject candidates in the same manner as we would have if we had used the true c.

If f and g are discrete distributions, then the proof of the ESUP algorithm is fairly straightforward. Specifically, Caffo, Booth, and Davison showed that P(Yi≠Y~i infinitely often)=0. Recall that by assumption, there exists some xc∈Xf such that c=f(xc)g(xc). Therefore, as we independently sample candidates from g, at some point, we will sample the value xc, in which case we will achieve the value c. Once that happens, we are then using the standard rejection sampling algorithm and our estimate c^ never changes.

Let γ=mini{xi=xc}, where xi∼g. So γ is the first time that we see the value xc as we are sampling candidates xi from g. The probability that we sample xc is g(xc) (recall that g is assumed to be discrete here) and so γ has a Geometric distribution with success probability g(xc). Once we observe xc, the ESUP algorithm and the standard rejection sampling algorithms converge and are identical.

From here, we can use the coupling inequality, which tells us that P(Yi≠Yi)≤P(γ≥i). Given that γ∼Geometric(g(xc)), we know that P(γ≥i)=(1−g(xc))i−1. This then implies that ∑i=1∞P(Yi≠Yi)<∞ which, by the Borel-Cantelli lemma, implies that P(Yi≠Yi infinitely often)=0. Therefore, eventually the sequences Yi and Yi must converge and at that point the ESUP algorithm will be identical to the rejection sampling algorithm.

In practice, we will know know exactly when the ESUP algorithm has converged to the standard rejection sampling algorithm. However, Caffo, and Davison report that the convergence is generally fast. Therefore, a reasonable approach might be to discard the first several accepted values (e.g. a “burn in”) and then use the remaining values.

We can see how quickly ESUP converges in a simple example where the target density is the standard Normal and the candidate density is the t2 distribution. Here we simulate 1,000 draws and start with a value c^=1.0001. Note that in the code below, all of the computations are done on the log scale for the sake of numerical stability.

<span id="cb122-1"><span>set.seed</span>(<span>2017-12-04</span>)</span>
<span id="cb122-2">N <span>&lt;-</span> <span>500</span></span>
<span id="cb122-3">y_tilde <span>&lt;-</span> <span>numeric</span>(N)  <span>## Binary accept/reject for "true" algorithm</span></span>
<span id="cb122-4">y <span>&lt;-</span> <span>numeric</span>(N)        <span>## Binary accept/reject for ESUP</span></span>
<span id="cb122-5">log_c_true <span>&lt;-</span> <span>dnorm</span>(<span>1</span>, <span>log =</span> <span>TRUE</span>) <span>-</span> <span>dt</span>(<span>1</span>, <span>2</span>, <span>log =</span> <span>TRUE</span>)</span>
<span id="cb122-6">log_chat <span>&lt;-</span> <span>numeric</span>(N <span>+</span> <span>1</span>)</span>
<span id="cb122-7">log_chat[<span>1</span>] <span>&lt;-</span> <span>log</span>(<span>1.0001</span>)  <span>## Starting c value</span></span>
<span id="cb122-8"><span>for</span>(i <span>in</span> <span>seq_len</span>(N)) {</span>
<span id="cb122-9">        u <span>&lt;-</span> <span>runif</span>(<span>1</span>)</span>
<span id="cb122-10">        x <span>&lt;-</span> <span>rt</span>(<span>1</span>, <span>2</span>)</span>
<span id="cb122-11">        r_true <span>&lt;-</span> <span>dnorm</span>(x, <span>log =</span> <span>TRUE</span>) <span>-</span> <span>dt</span>(x, <span>2</span>, <span>log =</span> <span>TRUE</span>) <span>-</span> log_c_true</span>
<span id="cb122-12">        rhat <span>&lt;-</span> <span>dnorm</span>(x, <span>log =</span> <span>TRUE</span>) <span>-</span> <span>dt</span>(x, <span>2</span>, <span>log =</span> <span>TRUE</span>) <span>-</span> log_chat[i]</span>
<span id="cb122-13">        y_tilde[i] <span>&lt;-</span> <span>log</span>(u) <span>&lt;=</span> r_true</span>
<span id="cb122-14">        y[i] <span>&lt;-</span> <span>log</span>(u) <span>&lt;=</span> rhat</span>
<span id="cb122-15">        log_chat[i<span>+</span><span>1</span>] <span>&lt;-</span> <span>max</span>(log_chat[i], </span>
<span id="cb122-16">                             <span>dnorm</span>(x, <span>log =</span> <span>TRUE</span>) <span>-</span> <span>dt</span>(x, <span>2</span>, <span>log =</span> <span>TRUE</span>))</span>
<span id="cb122-17">}</span>

Now we can plot log10⁡(|c^−c|) for each iteration to see how the magnitude of the error changes with each iteration.

<span id="cb123-1">c_true <span>&lt;-</span> <span>exp</span>(log_c_true)</span>
<span id="cb123-2">chat <span>&lt;-</span> <span>exp</span>(log_chat)</span>
<span id="cb123-3"><span>plot</span>(<span>log10</span>(<span>abs</span>(chat <span>-</span> c_true)), <span>type =</span> <span>"l"</span>,</span>
<span id="cb123-4">     <span>xlab =</span> <span>"Iteration"</span>, <span>ylab =</span> <span>expression</span>(<span>paste</span>(log[<span>10</span>], <span>"(Absolute Error)"</span>)))</span>

We can see that by iteration 40 or so, c^ and c differ only in the 5th decimal place and beyond. By the 380th iteration, they differ only beyond the 6th decimal place.