In my review of the paper Probabilistic forecasting of surgical case duration using machine learning: model development and validation last week, I briefly mentioned the term CRPS. Today, I'll go into a little more detail on what CRPS actually is and discuss its implementation in Python.
What is CRPS?
CRPS, which stands for Continous Ranked Probability Score, is a measure which is used to evaluate the result of a probabilistic distribution. I actually can't find where this term was created, but I have found references to CRPS dating back to at least 1977.
So why do we want to use CRPS? In normal regression problems, we can compare our prediction with the actual result using a simple formula like Mean Squared Error or Mean Absolute Error. However, in certain domains, our prediction isn't composed of just a single number, but a probabilistic distribution. Think about a prediction for how much snow we're going to get. Generally the weather forecasters won't say that we are going to get 8 inches of snow, they might say a 20% chance of 6 inches or less, 50% chance of 7-9 inches, and 30% chance of 10 inches or more. When the actual forecast comes in and we get 11 inches of snow, how do we evaluate this score?
This is where CRPS comes in, as it allows us to compare the actual result against the entire prediction distribution, instead of the just the mean of the distribution. The standard way we can calculate CRPS is with the integral
, where represents the Cumulative Density Function of our prediction distribution, and represents the point we are checking against.
While this equation is great, a severe limitation of this calculation is the inherent slowness of calculating this integral. To solve this issue, several researchers have come up with many closed-form solutions for certain common distributions. For example, here is the formula for a single normal distribution with mean and standard deviation , formulated by Gneiting et al in 2004:
Implementing CRPS in Python
If you want to use the CRPS scoring rule in Python, the first place you should look is the properscoring module. It provides a method, crps_quadrature()
, which allows users to numerically integrate any probability distribution function, given its CDF.
In addition, the module has also actually implemented the closed-form solution for the normal distribution, using the method crps_gaussian()
. With some quick time checks using both of these functions, crps_gaussian()
seems to run about 5000 times faster than integration on my machine, an absolutely enormous speedup!
However, there are a number of other closed-form functions which aren't implemented in properscoring. In this blog, I'll show my closed-form implementations of CRPS to be called for a normal distribution, a mixture of normal distributions, and a log-normal distribution.
Normal Distribution
def crps_norm(x, mu, sigma): sx = (x-mu)/sigma return sigma * (sx*(2*norm.cdf(sx)-1) + 2 * norm.pdf(sx) - 1/sqrt(pi))
Mixture of Normal Distributions
def crps_mixnorm(x, mus, sigmas, alphas): M = len(mus) def F(x): total = 0 for i in range(M): total += alphas[i] * norm.cdf((x-mus[i])/sigmas[i]) return total def A(mu1, mu2, sig1, sig2): sig = sqrt(sig1**2 + sig2**2) sx = (mu1-mu2)/sig return sx * sig * (2*norm.cdf(sx) - 1) + 2*sig*norm.pdf(sx) total = 0 for i in range(M): total += alphas[i] * A(x,mus[i], 0,sigmas[i]) for i in range(M): for j in range(M): total -= 0.5 * alphas[i] * alphas[j] * A(mus[i],mus[j], sigmas[i],sigmas[j]) return total
Log Normal Distribution
def crps_lognorm(x, mu, sigma): sx = (log(x)-mu)/sigma ex = 2 * exp(mu + sigma**2/2) return x * (2 * norm.cdf(sx) - 1) - ex * (norm.cdf(sx-sigma) + norm.cdf(sigma/sqrt(2)) - 1)
You can check the code at the bottom to view the results and confirm that these functions have been implemented correctly, and produce identical results to numeric integration. In fact, using the closed-form solutions is more accurate, due to the inherent rounding errors arising from integration.
Mixture of Log-Normal Distributions
While all of these implementations are great, the real reason I'm doing this is to use a closed-form solution for my own project, forecasting the duration of surgical operations. I was very excited about these implementations, so I could utilize the speedups in my own code. However, after doing some experiments, our distribution turns out to not be a mixture of normals, or a log-normal distribution, but a mixture of log-normals!
Unfortunately, no closed-form solution exists yet for this distribution. The closed form solution for mixture of log-normals was only derived very recently, in 2015, so it's possible that nobody has yet been interested enough in such a distribution. On the other hand, it's possible that the closed-form solution for the integration doesn't actually exist, which would be quite a bummer.
I spent a few days over the long weekend experimenting with existing solutions for other distributions to try and derive my own solution for the log-normal mixture. I believe that I've made some good progress, as I've found part of the solution, which, holding , , and constant, is only a constant term off from the real solution. I've verified this by cross-checking with the numeric-integration solutions, so this is a very good sign that the actual closed-form solution exists. Hopefully by the end of the rotation, I'll be able to derive the full solution!
Link to Jupyter Notebook
If you're interested, you can check out the Jupyter Notebook I've been working in, which contains all of the code I've written for this task, as well as results from all the experiments I've run so far.
1 Comment
Antonio · November 15, 2024 at 1:47pm
I believe the you have a small mistake on your CRPS formula. Most be- 1/sqrt(pi)