-
Notifications
You must be signed in to change notification settings - Fork 60
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Implemented model comparison. (#247)
* Implemented model comparison. * Improve documentation * Address comment * Relax the requirement for equal n_samples
- Loading branch information
Showing
7 changed files
with
98 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
"""This module contains methods for model comparison and selection.""" | ||
|
||
import numpy as np | ||
|
||
|
||
def compare_models(sample_objs, model_priors=None): | ||
"""Find posterior probabilities for different models. | ||
The algorithm requires elfi.Sample objects from prerun inference methods. For example the | ||
output from elfi.Rejection.sample is valid. The portion of samples for each model in the top | ||
discrepancies are adjusted by each models acceptance ratio and prior probability. | ||
The discrepancies (including summary statistics) must be comparable so that it is | ||
meaningful to sort them! | ||
Parameters | ||
---------- | ||
sample_objs : list of elfi.Sample | ||
Resulting Sample objects from prerun inference models. The objects must include | ||
a valid `discrepancies` attribute. | ||
model_priors : array_like, optional | ||
Prior probability of each model. Defaults to 1 / n_models. | ||
Returns | ||
------- | ||
np.array | ||
Posterior probabilities for the considered models. | ||
""" | ||
n_models = len(sample_objs) | ||
n_min = min([s.n_samples for s in sample_objs]) | ||
|
||
# concatenate discrepancy vectors | ||
try: | ||
discrepancies = np.concatenate([s.discrepancies for s in sample_objs]) | ||
except ValueError: | ||
raise ValueError("All Sample objects must include valid discrepancies.") | ||
|
||
# sort and take the smallest n_min | ||
inds = np.argsort(discrepancies)[:n_min] | ||
|
||
# calculate the portions of accepted samples for each model in the top discrepancies | ||
p_models = np.empty(n_models) | ||
up_bound = 0 | ||
for i in range(n_models): | ||
low_bound = up_bound | ||
up_bound += sample_objs[i].n_samples | ||
p_models[i] = np.logical_and(inds >= low_bound, inds < up_bound).sum() | ||
|
||
# adjust by the number of simulations run | ||
p_models[i] /= sample_objs[i].n_sim | ||
|
||
# adjust by the prior model probability | ||
if model_priors is not None: | ||
p_models[i] *= model_priors[i] | ||
|
||
p_models = p_models / p_models.sum() | ||
|
||
return p_models |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
import elfi | ||
from elfi.examples import gauss, ma2 | ||
|
||
|
||
def test_compare_models(): | ||
m = gauss.get_model() | ||
res1 = elfi.Rejection(m['d']).sample(100) | ||
|
||
# use less informative prior | ||
m['mu'].become(elfi.Prior('uniform', -10, 50)) | ||
res2 = elfi.Rejection(m['d']).sample(100) | ||
|
||
# use different simulator | ||
m['gauss'].become(elfi.Simulator(ma2.MA2, m['mu'], m['sigma'], observed=m.observed['gauss'])) | ||
res3 = elfi.Rejection(m['d']).sample(100) | ||
|
||
p = elfi.compare_models([res1, res2, res3]) | ||
assert p[0] > p[1] | ||
assert p[1] > p[2] |