LCOV - code coverage report
Current view: top level - memilio/epidemiology - lct_populations.h (source / functions) Hit Total Coverage
Test: coverage.info Lines: 29 29 100.0 %
Date: 2025-01-17 12:16:22 Functions: 163 175 93.1 %

          Line data    Source code
       1             : /* 
       2             : * Copyright (C) 2020-2025 MEmilio
       3             : *
       4             : * Authors: Lena Ploetzke
       5             : *
       6             : * Contact: Martin J. Kuehn <Martin.Kuehn@DLR.de>
       7             : *
       8             : * Licensed under the Apache License, Version 2.0 (the "License");
       9             : * you may not use this file except in compliance with the License.
      10             : * You may obtain a copy of the License at
      11             : *
      12             : *     http://www.apache.org/licenses/LICENSE-2.0
      13             : *
      14             : * Unless required by applicable law or agreed to in writing, software
      15             : * distributed under the License is distributed on an "AS IS" BASIS,
      16             : * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
      17             : * See the License for the specific language governing permissions and
      18             : * limitations under the License.
      19             : */
      20             : #ifndef MIO_EPI_LCT_POPULATIONS_H
      21             : #define MIO_EPI_LCT_POPULATIONS_H
      22             : 
      23             : #include "boost/type_traits/make_void.hpp"
      24             : #include "memilio/config.h"
      25             : #include "memilio/utils/uncertain_value.h"
      26             : #include "memilio/math/eigen.h"
      27             : #include "memilio/epidemiology/lct_infection_state.h"
      28             : #include "memilio/utils/type_list.h"
      29             : #include "memilio/utils/metaprogramming.h"
      30             : 
      31             : namespace mio
      32             : {
      33             : 
      34             : /**
      35             :  * @brief A class template for compartment populations of LCT models.
      36             :  *
      37             :  * Populations can be split up into different categories, e.g. by age group, yearly income group, gender etc. 
      38             :  * In LCT models, we want to use different numbers of subcompartments, i.e. different LctStates, 
      39             :  * for each group of a category.
      40             :  * (Therefore, we can't use the normal Populations class because it expects the same InfectionStates for each group.)
      41             :  * 
      42             :  * This template must be instantiated with one LctState for each group of a category. 
      43             :  * The purpose of the LctStates is to define the number of subcompartments for each InfectionState.
      44             :  * The number of LctStates also determines the number of groups. 
      45             :  * If you want to use more than one category, e.g. age and gender, you have to define num_age_groups * num_genders 
      46             :  * LctStates, because the number of subcompartments can be different 
      47             :  * even for (female, A05-A14) and (female, A80+) or (male, A05-A14).
      48             :  *
      49             :  * The class created from this template contains a "flat array" of compartment populations and some functions for 
      50             :  * retrieving or setting the populations. The order in the flat array is: First, all (sub-)compartments of the 
      51             :  * first group, afterwards all (sub-)compartments of the second group and so on.
      52             :  *
      53             :  */
      54             : 
      55             : template <typename FP = ScalarType, class... LctStates>
      56             : class LctPopulations
      57             : {
      58             : public:
      59             :     using Type                         = UncertainValue<FP>;
      60             :     using InternalArrayType            = Eigen::Array<Type, Eigen::Dynamic, 1>;
      61             :     using LctStatesGroups              = TypeList<LctStates...>;
      62             :     static size_t constexpr num_groups = sizeof...(LctStates); ///< Number of groups.
      63             :     static_assert(num_groups >= 1, "The number of LctStates provided should be at least one.");
      64             : 
      65             :     /// @brief Default constructor.
      66          21 :     LctPopulations()
      67          21 :     {
      68          21 :         set_count();
      69          21 :         m_y = InternalArrayType::Constant(m_count, 1, 0.);
      70          21 :     }
      71             : 
      72             :     /**
      73             :      * @brief get_num_compartments Returns the number of compartments.
      74             :      * @return Number of compartments which equals the flat array size.
      75             :      */
      76         879 :     size_t get_num_compartments() const
      77             :     {
      78         879 :         return m_count;
      79             :     }
      80             : 
      81             :     /**
      82             :      * @brief Returns a reference to the internally stored flat array.
      83             :      * @return Const reference to the InternalArrayType instance.
      84             :      */
      85             :     auto const& array() const
      86             :     {
      87             :         return m_y;
      88             :     }
      89             :     auto& array()
      90             :     {
      91             :         return m_y;
      92             :     }
      93             : 
      94             :     /**
      95             :      * @brief Returns the entry of the array given a flat index.
      96             :      * @param index A flat index.
      97             :      * @return The value of the internal array at the index.
      98             :      */
      99        2740 :     Type& operator[](size_t index)
     100             :     {
     101        2740 :         assert(index < m_count);
     102        2740 :         return m_y[index];
     103             :     }
     104             : 
     105             :     /**
     106             :     * @brief Gets the first index of a group in the flat array.
     107             :     * @tparam group The group for which the index should be returned.
     108             :     * @return The index of the first entry of group in the flat array.
     109             :     */
     110             :     template <size_t Group>
     111        2962 :     size_t get_first_index_of_group() const
     112             :     {
     113             :         static_assert((Group < num_groups) && (Group >= 0), "The template parameter Group should be valid.");
     114             :         if constexpr (Group == 0) {
     115        1597 :             return 0;
     116             :         }
     117             :         else {
     118        1365 :             return get_first_index_of_group<Group - 1>() + type_at_index_t<Group - 1, LctStatesGroups>::Count;
     119             :         }
     120             :     }
     121             :     /**
     122             :      * @brief Returns an Eigen copy of the vector of populations. 
     123             :      * This can be used as initial conditions for the ODE solver.
     124             :      * @return Eigen::VectorXd of populations.
     125             :      */
     126         201 :     inline Eigen::VectorX<FP> get_compartments() const
     127             :     {
     128         201 :         return m_y.array().template cast<FP>();
     129             :     }
     130             : 
     131             :     /**
     132             :      * @brief Returns the total population of a group.
     133             :      * @tparam group The group for which the total population should be calculated.
     134             :      * @return Total population of the group.
     135             :      */
     136             :     template <size_t Group>
     137           6 :     FP get_group_total() const
     138             :     {
     139           6 :         return m_y.array()
     140             :             .template cast<FP>()
     141           6 :             .segment(get_first_index_of_group<Group>(), type_at_index_t<Group, LctStatesGroups>::Count)
     142          18 :             .sum();
     143             :     }
     144             : 
     145             :     /**
     146             :      * @brief Returns the total population of all compartments and groups.
     147             :      * @return Total population.
     148             :      */
     149             :     FP get_total() const
     150             :     {
     151             :         return m_y.array().template cast<FP>().sum();
     152             :     }
     153             : 
     154             :     /**
     155             :      * @brief Checks whether all compartments have non-negative values. 
     156             :      * This function can be used to prevent slightly negative function values in compartment sizes that are produced 
     157             :      * due to rounding errors if, e.g., population sizes were computed in a complex way.
     158             :      *
     159             :      * Attention: This function should be used with care. It can not and will not set model parameters and 
     160             :      *            compartments to meaningful values. In most cases it is preferable to use check_constraints,
     161             :      *            and correct values manually before proceeding with the simulation.
     162             :      *            The main usage for apply_constraints is in automated tests using random values for initialization.
     163             :      *
     164             :      * @return Returns true if one (or more) constraint(s) were corrected, otherwise false.
     165             :     */
     166             :     bool apply_constraints()
     167             :     {
     168             :         bool corrected = false;
     169             :         for (int i = 0; i < m_y.array().size(); i++) {
     170             :             if (m_y.array()[i] < 0) {
     171             :                 log_warning("Constraint check: Compartment size {:d} changed from {:.4f} to {:d}", i, m_y.array()[i],
     172             :                             0);
     173             :                 m_y.array()[i] = 0.;
     174             :                 corrected      = true;
     175             :             }
     176             :         }
     177             :         return corrected;
     178             :     }
     179             : 
     180             :     /**
     181             :      * @brief Checks whether all compartments have non-negative values and logs an error if constraint is not satisfied.
     182             :      * @return Returns true if one or more constraints are not satisfied, false otherwise.
     183             :      */
     184           9 :     bool check_constraints() const
     185             :     {
     186           9 :         if ((m_y.array() < 0.).any()) {
     187           1 :             log_error("Constraint check: At least one compartment size is smaller {}.", 0);
     188           1 :             return true;
     189             :         }
     190           8 :         return false;
     191             :     }
     192             : 
     193             : private:
     194             :     /**
     195             :      * @brief Sets recursively the total number of (sub-)compartments over all groups.
     196             :      * The number also corresponds to the size of the internal vector.
     197             :      */
     198             :     template <size_t Group = 0>
     199          74 :     void set_count()
     200             :     {
     201             :         if constexpr (Group == 0) {
     202          21 :             m_count = 0;
     203             :         }
     204             :         if constexpr (Group < num_groups) {
     205          53 :             m_count += type_at_index_t<Group, LctStatesGroups>::Count;
     206          53 :             set_count<Group + 1>();
     207             :         }
     208          74 :     }
     209             : 
     210             :     size_t m_count; //< Number of groups stored.
     211             :     InternalArrayType m_y{}; //< An array containing the number of people in the groups.
     212             : };
     213             : 
     214             : } // namespace mio
     215             : 
     216             : #endif // MIO_EPI_LCT_POPULATIONS_H

Generated by: LCOV version 1.14