123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112 |
- // This file is part of BOINC.
- // http://boinc.berkeley.edu
- // Copyright (C) 2010 University of California
- //
- // BOINC is free software; you can redistribute it and/or modify it
- // under the terms of the GNU Lesser General Public License
- // as published by the Free Software Foundation,
- // either version 3 of the License, or (at your option) any later version.
- //
- // BOINC is distributed in the hope that it will be useful,
- // but WITHOUT ANY WARRANTY; without even the implied warranty of
- // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
- // See the GNU Lesser General Public License for more details.
- //
- // You should have received a copy of the GNU Lesser General Public License
- // along with BOINC. If not, see <http://www.gnu.org/licenses/>.
- // structure for tracking the recent average
- // of a distribution that may change over time
- //
- struct AVERAGE {
- double n; // double to avoid integer overflow
- double avg;
- inline void clear() {
- n = avg = 0;
- }
- // return true if sample exceeded limit and was truncated
- //
- inline bool update(
- double sample,
- double n_threshold,
- // after this many samples, use exponential average
- double sample_weight,
- // new samples get this weight in exp avg
- double sample_limit
- // truncate samples at avg*limit
- ) {
- double delta;
- bool truncated = false;
- if (sample < 0) return true;
- if (n && avg) {
- if (sample > avg*sample_limit) {
- sample = avg*sample_limit;
- truncated = true;
- }
- }
- n++;
- delta = sample - avg;
- if (n < n_threshold) {
- avg += delta/n;
- } else {
- avg += sample_weight*delta;
- }
- return truncated;
- }
- inline double get_avg() {
- return avg;
- }
- };
- // same, but variance also
- //
- struct AVERAGE_VAR : AVERAGE {
- double var;
- double q;
- inline bool update_var(
- double sample,
- double n_threshold,
- // after this many samples, use exponential average
- double sample_weight,
- // new samples get this weight in exp avg
- double sample_limit
- // truncate samples at avg*limit
- ) {
- double delta;
- bool truncated = false;
- if (sample < 0) return true;
- if (n && avg) {
- if (sample > avg*sample_limit) {
- sample = avg*sample_limit;
- truncated = true;
- }
- }
- n++;
- delta = sample - avg;
- if (n < n_threshold) {
- avg += delta/n;
- q += delta*(sample - avg);
- var = q/n;
- } else {
- avg += sample_weight*delta;
- double vdelta = (delta*delta - var);
- var += sample_weight*vdelta;
- }
- return truncated;
- }
- inline double get_var() {
- return var;
- }
- inline void clear() {
- AVERAGE::clear();
- var = 0;
- q = 0;
- }
- };
|