Likelihood Functions
In this lesson, we will have a look at Conditional Probabilities as well as Joint Probability.
In the previous lesson, we implemented an efficient conditioned probability using the Where
operator on distributions; that is, we have some “underlying” distribution, and we ask the question “if a particular condition has to be met, what is the derived distribution that meets that condition?” For discrete distributions, we can compute that distribution directly and just return it.
Conditional Probabilities as Likelihood Functions
There is another kind of conditional probability though, which is much more rich, complex and counter-intuitive, and that is exploring the relationship between “what is the probability of ?" and "what is the probability of given that we know ?
For example: pick a random person in the world who has a cold. What is the probability that they sneezed in the last hours? Probably something like .
Now pick a random person who does not have a cold. For them, the probability is maybe more like . In months when we do not have a cold, we sneeze maybe one or two days.
So what we’ve got here is a rather more complex probability distribution; in fact, we have two entirely different distributions, and which one we use depends on a condition.
Notice how this is related to our recent discussion of conditioned probabilities but different. With a Where
clause we are saying make the support of this distribution smaller because some outcomes are impossible based on a condition. What we’re talking about here is choosing between two (or more) distributions depending on a condition.
The standard notation for this kind of probability in mathematics is a bit unfortunate. We would say something like
to represent chance that we sneezed if we didn’t have a cold and
to represent chance that we sneezed if we had a cold. That is, the syntax is means what is the probability of given that happened?
How might we represent this in our system? It seems like IDiscreteDistribution<T>
is not rich enough. Let’s just start making some types and see what we can come up with.
“Has sneezed recently” and " has a cold" are Booleans, but we want the types of everything to be very clear in the analysis which follows, so we are going to make our custom types:
enum Cold { No, Yes }
enum Sneezed { No, Yes }
We want to be slightly abusive of notation here and say that P(Cold.Yes)
and P(Cold.No)
are the weights of a probability distribution that we are going to call by the shorthand P(Cold)
. Similarly for P(Sneezed)
; that’s the probability distribution that gives weights to P(Sneezed.Yes)
and P(Sneezed.No)
. Normally we think of P(something)
as being a value between and , but if you squint at it, really those values are just weights.
It doesn’t matter what convention we use for weights; a bunch of integers that give ratios of probabilities and a bunch of doubles that give fractions has pretty much the same information content.
Plainly what we would very much like is to have IDiscreteDistribution<Cold>
be the C#
type that represents P(Cold)
.
But how can we represent our concept of “There’s a chance we sneezed if we do not have a cold, but an chance if we do have a cold?”
That sure sounds like precisely this:
IDiscreteDistribution<Sneezed> SneezedGivenCold(Cold c)
{
var list = new List<Sneezed>() { Sneezed.No, Sneezed.Yes };
return c == Cold.No ? list.ToWeighted(97, 3) : list.ToWeighted(15, 85);
}
That is, if we do not have a cold then the odds are to that we did not sneeze, and if we do have a cold, then the odds are to that we did not sneeze.
We want to represent P(Cold.Yes)
and P(Cold.No)
by the shorthand P(Cold)
, and that this in our type system is IDiscreteDistribution<Cold>
. Now I want to represent the notion of P(Sneezed)
given a value of Cold as P(Sneezed|Cold)
, which is implemented by the function above. So, what type in our type system is that? Well, suppose we wanted to assign SneezedGivenCold
to a variable; what would its type be? Clearly, the type of P(Sneezed|Cold)
is Func<Cold, IDiscreteDistribution<Sneezed>>
!
How interesting! Conditional probabilities are actually functions.
This sort of function has a name; it is called a likelihood function. That is, given some condition, how likely is some outcome?
So that’s interesting, but how is this useful?
Example of a Likelihood Function
Let’s randomly choose a person in the world, where we do not know whether they have a cold or not. What is the probability that they sneezed recently? It depends entirely on the prevalence of colds! If of the world has a cold, then there’s an chance that a randomly chosen person sneezed recently, but if of the world has a cold, then there’s only a chance. And if it is somewhere in between, the probability will be different from either or .
To solve this problem we need to know the probability that the person we’ve chosen has a cold. The probability that a randomly chosen person has a cold is called the prior probability.
What if of the world has a cold? Let’s work it out by multiplying the probabilities:
Cold (prior) | Sneezed (likelihood) | Result (conditional) |
---|---|---|
Yes | Yes | have a cold, and sneezed |
No | have a cold, did not sneeze | |
No | Yes | do not have a cold and sneezed |
No | 87.3%$ do not have a cold, did not sneeze |
Sure enough, those probabilities in the right column add up to . The probability that a randomly chosen person in the world sneezed recently (given that these numbers that we made up are accurate) is .
The rightmost column of the table that we have sketched out here is called the joint probability, which we will notate as P(Cold&Sneezed)
.
Joint Probability
We can write this table more compactly like this:
Cold Yes | Cold No | Total | |
---|---|---|---|
Sneezed Yes | |||
Sneezed No | |||
Total |
The rightmost column of this table is called the marginal probability, so-called because of the way the sums end up at the margins of the table.
What if we expressed the marginal probability as integers? The odds that a random person has sneezed is to , which if you work out the math, is exactly odds of to .
represents the , not the .
How can we do this math given the set of types we’ve created so far? Let’s start with the prior:
var colds = new List<Cold>() { Cold.No, Cold.Yes };
IDiscreteDistribution<Cold> cold = colds.ToWeighted(90, 10);
We’ve got the prior, and we’ve got the likelihood function SneezedGivenCold
. We would like to get the marginal probability IDiscreteDistribution<Sneezed>
.
We could implement such a distribution by first sampling from the prior, then calling SneezedFromCold
, and then sampling from the returned distribution. Let’s implement it.
We are of course assuming that the likelihood function is pure.
public sealed class Combined<A, R> : IDiscreteDistribution<R>
{
private readonly List<R> support;
private readonly IDiscreteDistribution<A> prior;
private readonly Func<A, IDiscreteDistribution<R>> likelihood;
public static IDiscreteDistribution<R> Distribution(IDiscreteDistribution<A> prior, Func<A, IDiscreteDistribution<R>> likelihood) =>
new Combined<A, R>(prior, likelihood);
private Combined(IDiscreteDistribution<A> prior, Func<A, IDiscreteDistribution<R>> likelihood)
{
this.prior = prior;
this.likelihood = likelihood;
var q = from a in prior.Support()
from b in this.likelihood(a).Support()
select b;
this.support = q.Distinct().ToList();
}
public IEnumerable<R> Support() => this.support.Select(x => x);
public R Sample() => this.likelihood(this.prior.Sample()).Sample();
public int Weight(R r) => // WE’LL COME BACK TO THIS ONE
}
We haven’t implemented Weight
, but we don’t need it to run a histogram. Let’s try it out:
Combined<Cold, Sneezed>.Distribution(cold, SneezedGivenCold).Histogram()
The output will be:
No|****************************************
Yes|****
Sure enough, it looks like there is about an chance that a randomly chosen person sneezed, given these distributions.
Now, of course as we have done throughout this series, let’s make a little helper function to make the call sites look a little nicer:
public static IDiscreteDistribution<R> MakeCombined<A, R>(this IDiscreteDistribution<A> prior, Func<A, IDiscreteDistribution<R>> likelihood) =>
Combined<A, R>.Distribution(prior, likelihood);
Once again, that should look very familiar! We should change the name of this helper.
If you are still surprised at this point, you have not been paying attention. I’ve already made Select
and Where
, so the obvious next step is…
public static IDiscreteDistribution<R> SelectMany<A, R>(this IDiscreteDistribution<A> prior, Func<A, IDiscreteDistribution<R>> likelihood) =>
Combined<A, R>.Distribution(prior, likelihood);
… the bind operation on the probability monad.
And the inelegant call site above is now the much more clear:
cold.SelectMany(SneezedGivenCold).Histogram()
Implementation
The code for this lesson is as follows:
Get hands-on with 1400+ tech skills courses.