Subversion Repositories public iLand

Rev

Rev 391 | Rev 420 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
1
 
90 Werner 2
#ifndef SPECIES_H
3
#define SPECIES_H
38 Werner 4
 
103 Werner 5
 
91 Werner 6
#include "expression.h"
7
 
103 Werner 8
#include "speciesset.h"
102 Werner 9
 
91 Werner 10
class StampContainer; // forwards
38 Werner 11
class Stamp;
91 Werner 12
 
103 Werner 13
 
90 Werner 14
class Species
38 Werner 15
{
16
public:
387 werner 17
    Species(SpeciesSet *set) { mSet = set; mIndex=set->count(); mSeedDispersal=0; }
391 werner 18
    ~Species();
19
    // maintenance
20
    void setup();
415 werner 21
    void newYear();
391 werner 22
 
226 werner 23
    const SpeciesSet *speciesSet() const { return mSet; }
91 Werner 24
    // properties
391 werner 25
    SeedDispersal *seedDispersal() const { return mSeedDispersal; }
91 Werner 26
    /// @property id 4-character unique identification of the tree species
111 Werner 27
    const QString &id() const { return mId; }
91 Werner 28
    /// the full name (e.g. Picea Abies) of the species
111 Werner 29
    const QString &name() const { return mName; }
145 Werner 30
    int index() const { return mIndex; } ///< unique index of species within current set
179 werner 31
    bool active() const { return true; } ///< active??? todo!
236 werner 32
    int phenologyClass() const { return mPhenologyClass; } ///< phenology class defined in project file. class 0 = evergreen
33
    bool isConiferous() const { return mConiferous; }
34
    bool isEvergreen() const { return mEvergreen; }
415 werner 35
    bool isSeedYear() const { return mIsSeedYear; }
136 Werner 36
 
391 werner 37
 
91 Werner 38
    // calculations: allometries
145 Werner 39
    double biomassFoliage(const double dbh) const;
40
    double biomassWoody(const double dbh) const;
41
    double biomassRoot(const double dbh) const;
42
    double allometricRatio_wf() const { return mWoody_b / mFoliage_b; }
43
    double allometricFractionStem(const double dbh) const;
276 werner 44
    double finerootFoliageRatio() const { return mFinerootFoliageRatio; } ///< ratio of fineroot mass (kg) to foliage mass (kg)
136 Werner 45
 
116 Werner 46
    // turnover rates
145 Werner 47
    double turnoverLeaf() const { return mTurnoverLeaf; }
48
    double turnoverRoot() const { return mTurnoverRoot; }
119 Werner 49
    // hd-values
50
    void hdRange(const double dbh, double &rMinHD, double &rMaxHD);
125 Werner 51
    // growth
145 Werner 52
    double volumeFactor() const { return mVolumeFactor; } ///< factor for volume calculation: V = factor * D^2*H (incorporates density and the form of the bole)
53
    double density() const { return mWoodDensity; } ///< density of stem wood [kg/m3]
54
    double specificLeafArea() const { return mSpecificLeafArea; }
159 werner 55
    // mortality
56
    double deathProb_intrinsic() const { return mDeathProb_intrinsic; }
308 werner 57
    inline double deathProb_stress(const double &stress_index) const;
169 werner 58
    // aging
388 werner 59
    double aging(const float height, const int age);
60
    int estimateAge(const float height) const;///< estimate age for a tree with the current age
387 werner 61
    // regeneration
62
    void seedProduction(const float height, const QPoint &position_index);
63
    void setSeedDispersal(SeedDispersal *seed_dispersal) {mSeedDispersal=seed_dispersal; }
209 werner 64
    // environmental responses
65
    double vpdResponse(const double &vpd) const;
266 werner 66
    inline double temperatureResponse(const double &delayed_temp) const;
209 werner 67
    double nitrogenResponse(const double &availableNitrogen) const { return mSet->nitrogenResponse(availableNitrogen, mRespNitrogenClass); }
236 werner 68
    double canopyConductance() const { return mMaxCanopyConductance; } ///< maximum canopy conductance in m/s
266 werner 69
    inline double soilwaterResponse(const double &psi_kPa) const; ///< input: matrix potential (kPa) (e.g. -15)
274 werner 70
    double lightResponse(const double lightResourceIndex) {return mSet->lightResponse(lightResourceIndex, mLightResponseClass); }
304 werner 71
    double psiMin() const { return mPsiMin; }
110 Werner 72
 
136 Werner 73
    const Stamp* stamp(const float dbh, const float height) const { return mLIPs.stamp(dbh, height);}
38 Werner 74
private:
90 Werner 75
    Q_DISABLE_COPY(Species);
265 werner 76
    QMutex mMutex;
136 Werner 77
    // helpers during setup
236 werner 78
    bool boolVar(const QString s) { return mSet->var(s).toBool(); } ///< during setup: get value of variable @p s as a boolean variable.
136 Werner 79
    double doubleVar(const QString s) { return mSet->var(s).toDouble(); }///< during setup: get value of variable @p s as a double.
236 werner 80
    int intVar(const QString s) { return mSet->var(s).toInt(); } ///< during setup: get value of variable @p s as an integer.
136 Werner 81
    QString stringVar(const QString s) { return mSet->var(s).toString(); } ///< during setup: get value of variable @p s as a string.
82
 
91 Werner 83
    SpeciesSet *mSet; ///< ptr. to the "parent" set
136 Werner 84
    StampContainer mLIPs; ///< ptr to the container of the LIP-pattern
91 Werner 85
    QString mId;
86
    QString mName;
111 Werner 87
    int mIndex; ///< internal index within the SpeciesSet
236 werner 88
    bool mConiferous; ///< true if confierous species (vs. broadleaved)
89
    bool mEvergreen; ///< true if evergreen species
136 Werner 90
    // biomass allometries:
91
    double mFoliage_a, mFoliage_b;  ///< allometry (biomass = a * dbh^b) for foliage
92
    double mWoody_a, mWoody_b; ///< allometry (biomass = a * dbh^b) for woody compartments aboveground
93
    double mRoot_a, mRoot_b; ///< allometry (biomass = a * dbh^b) for roots (compound, fine and coarse roots as one pool)
94
    double mBranch_a, mBranch_b; ///< allometry (biomass = a * dbh^b) for branches
95
 
110 Werner 96
    double mSpecificLeafArea; ///< conversion factor from kg OTS to m2 LeafArea
116 Werner 97
    // turnover rates
98
    double mTurnoverLeaf; ///< yearly turnover rate leafs
99
    double mTurnoverRoot; ///< yearly turnover rate root
276 werner 100
    double mFinerootFoliageRatio; ///< ratio of fineroot mass (kg) to foliage mass (kg)
119 Werner 101
    // height-diameter-relationships
102
    Expression mHDlow; ///< minimum HD-relation as f(d) (open grown tree)
103
    Expression mHDhigh; ///< maximum HD-relation as f(d)
125 Werner 104
    // stem density and taper
105
    double mWoodDensity; ///< density of the wood [kg/m3]
106
    double mFormFactor; ///< taper form factor of the stem [-] used for volume / stem-mass calculation calculation
107
    double mVolumeFactor; ///< factor for volume calculation
159 werner 108
    // mortality
109
    double mDeathProb_intrinsic;  ///< prob. of intrinsic death per year [0..1]
110
    double mDeathProb_stress; ///< max. prob. of death per year when tree suffering maximum stress
169 werner 111
    // Aging
112
    double mMaximumAge; ///< maximum age of species (years)
113
    double mMaximumHeight; ///< maximum height of species (m) for aging
214 werner 114
    Expression mAging;
209 werner 115
    // environmental responses
116
    double mRespVpdExponent; ///< exponent in vpd response calculation (Mäkela 2008)
117
    double mRespTempMin; ///< temperature response calculation offset
118
    double mRespTempMax; ///< temperature response calculation: saturation point for temp. response
119
    double mRespNitrogenClass; ///< nitrogen response class (1..3). fractional values (e.g. 1.2) are interpolated.
304 werner 120
    double mPsiMin; ///< minimum water potential (MPa), i.e. wilting point (is below zero!)
236 werner 121
    // water
122
    double mMaxCanopyConductance; ///< maximum canopy conductance for transpiration (m/s)
226 werner 123
    int mPhenologyClass;
274 werner 124
    double mLightResponseClass; ///< light response class (1..5) (1=shade intolerant)
387 werner 125
    // regeneration
126
    SeedDispersal *mSeedDispersal; ///< link to the seed dispersal map of the species
415 werner 127
    double mSeedYearProbability; ///< probability that a year is a seed year (=1/avg.timespan between seedyears)
128
    bool mIsSeedYear; ///< true, if current year is a seed year. see also:
38 Werner 129
};
130
 
40 Werner 131
 
119 Werner 132
// inlined functions...
133
inline void Species::hdRange(const double dbh, double &rLowHD, double &rHighHD)
134
{
265 werner 135
    QMutexLocker m(&mMutex); // serialize access
119 Werner 136
    rLowHD = mHDlow.calculate(dbh);
137
    rHighHD = mHDhigh.calculate(dbh);
138
}
209 werner 139
/** vpdResponse calculates response on vpd.
140
    Input: vpd [kPa]*/
141
inline double Species::vpdResponse(const double &vpd) const
142
{
143
    return exp(mRespVpdExponent * vpd);
144
}
119 Werner 145
 
209 werner 146
/** temperatureResponse calculates response on delayed daily temperature.
147
    Input: average temperature [°C]
148
    Note: slightly different from Mäkela 2008: the maximum parameter (Sk) in iLand is interpreted as the absolute
149
          temperature yielding a response of 1; in Mäkela 2008, Sk is the width of the range (relative to the lower threhold)
150
*/
151
inline double Species::temperatureResponse(const double &delayed_temp) const
152
{
153
    double x = qMax(delayed_temp-mRespTempMin, 0.);
154
    x = qMin(x/(mRespTempMax-mRespTempMin), 1.);
155
    return x;
156
}
266 werner 157
/** soilwaterResponse is a function of the current matrix potential of the soil.
209 werner 158
 
266 werner 159
  */
160
inline double Species::soilwaterResponse(const double &psi_kPa) const
161
{
162
    const double psi_mpa = psi_kPa / 1000.; // convert to MPa
304 werner 163
    double result = limit( 1. - psi_mpa / mPsiMin, 0., 1.);
266 werner 164
    return result;
165
}
166
 
308 werner 167
/** calculate probabilty of death based on the current stress index. */
168
inline double Species::deathProb_stress(const double &stress_index) const
169
{
170
    if (stress_index==0)
171
        return 0.;
172
    double result = 1. - exp(-mDeathProb_stress*stress_index);
173
    return result;
174
}
175
 
90 Werner 176
#endif // SPECIES_H