IODA
Accumulator.h
Go to the documentation of this file.
1 /*
2  * (C) Crown copyright 2021, Met Office
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  */
7 
8 #ifndef DISTRIBUTION_ACCUMULATOR_H_
9 #define DISTRIBUTION_ACCUMULATOR_H_
10 
11 #include <vector>
12 
13 namespace ioda {
14 
15 /// \brief Calculates the sum of a location-dependent quantity of type `T` over locations held on
16 /// all PEs, each taken into account only once even if it's held on multiple PEs.
17 ///
18 /// The intended usage is as follows:
19 /// 1. Create an Accumulator by calling the `createAccumulator()` method of a Distribution.
20 /// 2. Iterate over locations held on the current PE and call addTerm() for each location making a
21 /// non-zero contribution to the sum.
22 /// 3. Call computeResult() to calculate the global sum (over all PEs).
23 ///
24 /// Subclasses need to implement addTerm() and computeResult() in such a way that contributions made
25 /// by locations held on multiple PEs are included only once in the global sum.
26 template <typename T>
27 class Accumulator {
28  public:
29  virtual ~Accumulator() {}
30 
31  /// \brief Increment the sum with the contribution `term` of location `loc` held on the
32  /// current PE.
33  virtual void addTerm(std::size_t loc, const T & term) = 0;
34 
35  /// \brief Return the sum of contributions associated with locations held on all PEs
36  /// (each taken into account only once).
37  virtual T computeResult() const = 0;
38 };
39 
40 /// \brief Calculates the sums of multiple location-dependent quantities of type `T` over locations
41 /// held on all PEs, each taken into account only once even if it's held on multiple PEs.
42 ///
43 /// The intended usage is the same as of the primary template, except that this specialization for
44 /// `std::vector<T>` provides two overloads of addTerm(). Use whichever is more convenient.
45 template <typename T>
46 class Accumulator<std::vector<T>> {
47  public:
48  virtual ~Accumulator() {}
49 
50  /// \brief Increment each sum with the contribution of location `loc` (held on the current PE)
51  /// taken from the corresponding element of `term`.
52  virtual void addTerm(std::size_t loc, const std::vector<T> & term) = 0;
53 
54  /// \brief Increment the `i`th sum with the contribution `term` of location `loc` held on the
55  /// current PE.
56  virtual void addTerm(std::size_t loc, std::size_t i, const T & term) = 0;
57 
58  /// \brief Return the sums of contributions associated with locations held on all
59  /// PEs (each taken into account only once).
60  virtual std::vector<T> computeResult() const = 0;
61 };
62 
63 } // namespace ioda
64 
65 #endif // DISTRIBUTION_ACCUMULATOR_H_
virtual void addTerm(std::size_t loc, std::size_t i, const T &term)=0
Increment the ith sum with the contribution term of location loc held on the current PE.
virtual void addTerm(std::size_t loc, const std::vector< T > &term)=0
Increment each sum with the contribution of location loc (held on the current PE) taken from the corr...
virtual std::vector< T > computeResult() const =0
Return the sums of contributions associated with locations held on all PEs (each taken into account o...
Calculates the sum of a location-dependent quantity of type T over locations held on all PEs,...
Definition: Accumulator.h:27
virtual T computeResult() const =0
Return the sum of contributions associated with locations held on all PEs (each taken into account on...
virtual ~Accumulator()
Definition: Accumulator.h:29
virtual void addTerm(std::size_t loc, const T &term)=0
Increment the sum with the contribution term of location loc held on the current PE.