File: LimitedScan.cpp

package info (click to toggle)
ausaxs 1.1.8-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 72,592 kB
  • sloc: cpp: 49,853; ansic: 6,901; python: 730; makefile: 18
file content (80 lines) | stat: -rw-r--r-- 2,665 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
// SPDX-License-Identifier: LGPL-3.0-or-later
// Author: Kristian Lytje

#include <mini/LimitedScan.h>
#include <utility/Limit.h>
#include <mini/detail/Parameter.h>

#include <limits>
#include <list>
#include <numeric>

using namespace ausaxs;
using namespace ausaxs::mini;

LimitedScan::~LimitedScan() = default;

void LimitedScan::set_limit(double limit, bool minima_multiplier) noexcept {
    this->limit = limit;
    this->limit_is_minima_multiplier = minima_multiplier;
}

mini::Landscape LimitedScan::landscape(unsigned int evals) {
    // check if the minimizer has already been called
    if (!evaluations.evals.empty()) {
        return evaluations; // if so, we can just reuse its result
    }

    double current_min = std::numeric_limits<double>::max();
    std::function<bool(double)> stop_condition;
    if (limit_is_minima_multiplier) {
        stop_condition = [&](double val) {
            return current_min*limit < val;
        };
    } else {
        stop_condition = [&](double val) {
            return limit < val;
        };
    }

    if (parameters.size() == 1) {
        const Limit& bounds = parameters[0].bounds.value();
        unsigned int c = 0;
        std::list<double> last_evals;
        unsigned int count = 0;
        for (double val = bounds.max; bounds.min < val; val -= bounds.span()/evals) {
            double fval = function({val});
            current_min = std::min(current_min, fval);

            // add the evaluation to the list
            if (last_evals.size() < 7) {
                last_evals.push_front(fval);
            } else {
                last_evals.pop_back();
                last_evals.push_front(fval);
            }
            
            // calculate average of list
            double avg = std::accumulate(last_evals.begin(), last_evals.end(), 0.0) / last_evals.size();

            // if we are more than half-way through the scan, we check for the stop condition
            if (evals*0.7 < ++c) {
                // if the fval is greater than the limit and we are not converging on a solution, we stop
                if (stop_condition(avg) && stop_condition(fval)) {
                    ++count;
                    // three consecutive fvals greater than the avg also means we stop
                    if (3 == count) {
                        break;
                    }
                } else {
                    count = 0;
                }
            }
        }
        return get_evaluated_points();
    }

    else { // parameters.size() <= 2 
        throw std::runtime_error("LimitedScan::landscape: Using more than two parameters is currently not implemented.");
    } 
}