2020-03-06 02:24:32 +01:00
|
|
|
#include <gsl/gsl_monte_plain.h>
|
|
|
|
#include <gsl/gsl_monte_miser.h>
|
|
|
|
#include <gsl/gsl_monte_vegas.h>
|
|
|
|
#include <gsl/gsl_monte.h>
|
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
|
2020-05-24 22:04:31 +02:00
|
|
|
// exp() wrapper for the gsl_function structure
|
2020-05-15 00:07:04 +02:00
|
|
|
//
|
|
|
|
double function (double * x, size_t dim, void * params)
|
|
|
|
{
|
2020-03-06 02:24:32 +01:00
|
|
|
return exp(x[0]);
|
2020-05-15 00:07:04 +02:00
|
|
|
}
|
2020-03-06 02:24:32 +01:00
|
|
|
|
|
|
|
|
|
|
|
// Function which prints out the results when called.
|
2020-05-15 00:07:04 +02:00
|
|
|
//
|
2020-05-19 16:01:52 +02:00
|
|
|
void results (size_t calls, double result, double error, double chi)
|
2020-05-15 00:07:04 +02:00
|
|
|
{
|
|
|
|
if (calls != 0)
|
2020-04-30 22:25:44 +02:00
|
|
|
{
|
2020-05-24 22:04:31 +02:00
|
|
|
printf ("%6.0e | ", (double)calls);
|
2020-04-30 22:25:44 +02:00
|
|
|
}
|
2020-05-24 22:04:31 +02:00
|
|
|
printf ("%5f | ", result);
|
|
|
|
printf ("%5f | ", error);
|
2020-05-19 16:01:52 +02:00
|
|
|
if (chi != 0)
|
|
|
|
{
|
2020-05-24 22:04:31 +02:00
|
|
|
printf ("%5f", chi);
|
2020-05-19 16:01:52 +02:00
|
|
|
}
|
2020-05-15 00:07:04 +02:00
|
|
|
}
|
|
|
|
|
2020-04-30 22:25:44 +02:00
|
|
|
|
2020-05-15 00:07:04 +02:00
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
2020-03-06 02:24:32 +01:00
|
|
|
// Some useful variables.
|
2020-05-15 00:07:04 +02:00
|
|
|
//
|
|
|
|
size_t dims = 1; // Integral dimension
|
2020-03-06 02:24:32 +01:00
|
|
|
double lower[1] = {0}; // Integration lower limit
|
|
|
|
double upper[1] = {1}; // Integration upper limit
|
2020-05-19 16:01:52 +02:00
|
|
|
size_t calls = 50; // Initial number of function calls
|
2020-05-15 00:07:04 +02:00
|
|
|
double integral, error;
|
2020-03-06 02:24:32 +01:00
|
|
|
|
|
|
|
// Initialize an RNG.
|
2020-05-15 00:07:04 +02:00
|
|
|
//
|
2020-03-06 02:24:32 +01:00
|
|
|
gsl_rng_env_setup();
|
|
|
|
gsl_rng *r = gsl_rng_alloc(gsl_rng_default);
|
|
|
|
|
|
|
|
// Define GSL function.
|
2020-05-15 00:07:04 +02:00
|
|
|
//
|
2020-03-06 02:24:32 +01:00
|
|
|
gsl_monte_function expo = {&function, dims, NULL};
|
|
|
|
expo.f = &function;
|
|
|
|
expo.dim = 1;
|
|
|
|
expo.params = NULL;
|
|
|
|
|
2020-05-24 22:04:31 +02:00
|
|
|
// Print the results table header
|
|
|
|
printf(" Calls | Plain | Error | MISER |"
|
|
|
|
" Error | VEGAS | Error | χ²\n");
|
|
|
|
printf(" ------|----------|----------|----------|"
|
|
|
|
"----------|----------|----------|---------\n");
|
|
|
|
|
2020-05-15 00:07:04 +02:00
|
|
|
// Compute the integral for the following number of function calls:
|
|
|
|
// 50
|
|
|
|
// 500
|
|
|
|
// 5'000
|
|
|
|
// 50'000
|
|
|
|
// 500'000
|
|
|
|
// 5'000'000
|
|
|
|
// 50'000'000
|
|
|
|
// with the three MC methods: plain MC, MISER and VEGAS.
|
|
|
|
//
|
2020-05-24 22:04:31 +02:00
|
|
|
while (calls <= 5000000)
|
2020-05-15 00:07:04 +02:00
|
|
|
{
|
|
|
|
// Plain MC
|
|
|
|
gsl_monte_plain_state *sMC = gsl_monte_plain_alloc (dims);
|
|
|
|
gsl_monte_plain_integrate (&expo, lower, upper, dims, calls,
|
|
|
|
r, sMC, &integral, &error);
|
|
|
|
gsl_monte_plain_free(sMC);
|
2020-05-19 16:01:52 +02:00
|
|
|
results(calls, integral, error, 0);
|
2020-03-06 02:24:32 +01:00
|
|
|
|
2020-05-15 00:07:04 +02:00
|
|
|
// MISER
|
|
|
|
gsl_monte_miser_state *sMI = gsl_monte_miser_alloc (dims);
|
2020-03-06 02:24:32 +01:00
|
|
|
gsl_monte_miser_integrate (&expo, lower, upper, dims, calls,
|
2020-05-15 00:07:04 +02:00
|
|
|
r, sMI, &integral, &error);
|
|
|
|
gsl_monte_miser_free(sMI);
|
2020-05-19 16:01:52 +02:00
|
|
|
results(0, integral, error, 0);
|
2020-03-06 02:24:32 +01:00
|
|
|
|
2020-05-15 00:07:04 +02:00
|
|
|
// VEGAS
|
|
|
|
gsl_monte_vegas_state *sVE = gsl_monte_vegas_alloc (dims);
|
|
|
|
gsl_monte_vegas_integrate (&expo, lower, upper, dims, calls,
|
|
|
|
r, sVE, &integral, &error);
|
2020-05-24 22:04:31 +02:00
|
|
|
|
2020-05-19 16:01:52 +02:00
|
|
|
// In order to see the parameters of the VEGAS integration, decomment this
|
|
|
|
// paragraph:
|
|
|
|
//
|
|
|
|
// gsl_monte_vegas_params params;
|
|
|
|
// gsl_monte_vegas_params_get(sVE, ¶ms);
|
|
|
|
// params.verbose = 1;
|
|
|
|
// params.ostream = stderr;
|
|
|
|
// gsl_monte_vegas_params_set(sVE, ¶ms);
|
2020-05-15 00:07:04 +02:00
|
|
|
do
|
|
|
|
{
|
|
|
|
gsl_monte_vegas_integrate (&expo, lower, upper, dims,
|
2020-05-19 16:01:52 +02:00
|
|
|
calls, r, sVE, &integral, &error);
|
2020-05-15 00:07:04 +02:00
|
|
|
} while (fabs (gsl_monte_vegas_chisq (sVE) - 1.0) > 0.5);
|
|
|
|
gsl_monte_vegas_free(sVE);
|
2020-05-19 16:01:52 +02:00
|
|
|
double chi = sVE->chisq;
|
|
|
|
results(0, integral, error, chi);
|
|
|
|
|
2020-05-15 00:07:04 +02:00
|
|
|
|
|
|
|
// Update function calls
|
|
|
|
printf ("\n");
|
|
|
|
calls = calls*10;
|
|
|
|
}
|
2020-03-06 02:24:32 +01:00
|
|
|
|
|
|
|
// Free memory.
|
|
|
|
gsl_rng_free(r);
|
|
|
|
|
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|