Subversion Repositories public iLand

Rev

Rev 1221 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
671 werner 1
/********************************************************************************************
2
**    iLand - an individual based forest landscape and disturbance model
3
**    http://iland.boku.ac.at
4
**    Copyright (C) 2009-  Werner Rammer, Rupert Seidl
5
**
6
**    This program is free software: you can redistribute it and/or modify
7
**    it under the terms of the GNU General Public License as published by
8
**    the Free Software Foundation, either version 3 of the License, or
9
**    (at your option) any later version.
10
**
11
**    This program is distributed in the hope that it will be useful,
12
**    but WITHOUT ANY WARRANTY; without even the implied warranty of
13
**    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
**    GNU General Public License for more details.
15
**
16
**    You should have received a copy of the GNU General Public License
17
**    along with this program.  If not, see <http://www.gnu.org/licenses/>.
18
********************************************************************************************/
19
 
211 werner 20
#include "global.h"
21
#include "phenology.h"
22
 
23
#include "climate.h"
214 werner 24
#include "floatingaverage.h"
25
 
26
/// "ramp" function: value < min: 0, value>max: 1: in between linear interpolation
27
double ramp(const double &value, const double minValue, const double maxValue)
28
{
29
    Q_ASSERT(minValue!=maxValue);
30
    if (value<minValue) return 0.;
31
    if (value>maxValue) return 1.;
32
    return (value-minValue) / (maxValue - minValue);
33
}
34
 
697 werner 35
/** @class Phenology phenology submodule.
36
  @ingroup core
37
  The Phenology submodule calculates the length of the growing season according to the model of Jolly et al (2005). The calculation
38
  is performed for species-groups (i.e.: species are lumped together to groups) and a given climate (i.e. worst case: for each ResourceUnit).
39
 
40
  See http://iland.boku.ac.at/phenology for details.
41
  */
42
 
214 werner 43
/** calculates the phenology according to Jolly et al. 2005.
226 werner 44
  The calculation is performed for a given "group" and a present "climate".
214 werner 45
*/
46
void Phenology::calculate()
47
{
436 werner 48
    if (id()==0) {
49
        // for needles: just calculate the chilling requirement for the establishment
440 werner 50
        // i.e.: use the "bottom line" of 10.5 hrs daylength for the end of the vegetation season
51
        calculateChillDays(mClimate->sun().dayShorter10_5hrs());
214 werner 52
        return;
436 werner 53
    }
214 werner 54
    const ClimateDay  *end = mClimate->end();
55
    double vpd,temp,daylength;
56
    double gsi; // combined factor of effect of vpd, temperature and day length
57
    int iday = 0;
58
    bool inside_period = !mClimate->sun().northernHemishere(); // on northern hemisphere 1.1. is in winter
59
    int day_start=-1, day_stop=-1;
60
    int day_wait_for=-1;
61
 
62
    FloatingAverage floater(21); // a three-week floating average
63
    for (const ClimateDay *day = mClimate->begin(); day!=end; ++day, ++iday) {
64
 
65
        if (day_wait_for>=0 && iday<day_wait_for)
66
            continue;
67
        vpd = 1. - ramp(day->vpd, mMinVpd, mMaxVpd); // high value for low vpd
441 werner 68
        temp = ramp(day->min_temperature, mMinTemp, mMaxTemp);
214 werner 69
        daylength = ramp( mClimate->sun().daylength(iday), mMinDayLength, mMaxDayLength);
70
        gsi = vpd * temp * daylength;
71
        gsi = floater.add(gsi);
72
        if (!inside_period && gsi>0.5) {
73
            // switch from winter -> summer
74
            inside_period = true;
75
            day_start = iday;
76
            if (day_stop!=-1)
77
                break;
78
            day_wait_for = mClimate->sun().longestDay();
79
        } else if (inside_period && gsi<0.5) {
80
            // switch from summer to winter
81
            day_stop = iday;
82
            if (day_start!=-1)
83
                break; // finished
84
            day_wait_for = mClimate->sun().longestDay();
85
            inside_period = false;
86
        }
87
    }
88
    day_start -= 10; // three-week-floating average: subtract 10 days
89
    day_stop -= 10;
1103 werner 90
    if (day_start < -1 || day_stop<-1) {
91
        //throw IException(QString("Phenology::calculation(): was not able to determine the length of the vegetation period for group %1. climate table: '%2'." ).arg(id()).arg(mClimate->name()));
92
        qDebug() << "Phenology::calculation(): vegetation period is 0 for group" << id() << ", climate table: " << mClimate->name();
93
        day_start = mClimate->daysOfYear()-1; // last day of the year, never reached
94
        day_stop = day_start; // never reached
95
    }
611 werner 96
    if (logLevelDebug())
97
        qDebug() << "Jolly-phenology. start" << mClimate->dayOfYear(day_start)->toString() << "stop" << mClimate->dayOfYear(day_stop)->toString();
214 werner 98
    iday = 0;
226 werner 99
    mDayStart = day_start;
100
    mDayEnd = day_stop;
214 werner 101
    int bDay, bMon, eDay, eMon;
102
    // convert yeardays to dates
103
    mClimate->toDate(day_start, &bDay, &bMon);
104
    mClimate->toDate(day_stop, &eDay, &eMon);
105
    for (int i=0;i<12;i++) {
106
        if (i<bMon || i>eMon) {
107
            mPhenoFraction[i] = 0; // out of season
108
        } else if (i>bMon && i<eMon) {
109
            mPhenoFraction[i] = 1.; // full inside of season
110
        } else {
111
            // fractions of month
112
            mPhenoFraction[i] = 1.;
113
            if (i==bMon)
114
                mPhenoFraction[i] -=  (bDay+1) / double(mClimate->days(bMon));
115
            if (i==eMon)
116
                mPhenoFraction[i] -=  (mClimate->days(eMon) - (bDay+1)) / double(mClimate->days(eMon));
117
        }
118
    }
434 werner 119
 
120
    calculateChillDays();
214 werner 121
}
434 werner 122
 
123
 
124
// *********** Chill-day calculations ********
436 werner 125
void Phenology::calculateChillDays(const int end_of_season)
434 werner 126
{
127
    int iday = 0;
128
    mChillDaysBefore = 0;
129
    int days_after = 0;
436 werner 130
    int last_day = end_of_season>0?end_of_season:mDayEnd;
434 werner 131
    for (const ClimateDay *day = mClimate->begin(); day!=mClimate->end(); ++day, ++iday) {
720 werner 132
        if (day->temperature>=-5. && day->temperature<5.) {
434 werner 133
            if (iday<mDayStart)
134
                mChillDaysBefore++;
436 werner 135
            if (iday>last_day)
434 werner 136
                days_after++;
137
        }
138
    }
139
    if (GlobalSettings::instance()->currentYear()==1) {
140
        // for the first simulation year, use the value of this autumn for the last years autumn
141
        mChillDaysAfterLastYear = days_after;
142
    } else {
143
        mChillDaysAfterLastYear  = mChillDaysAfter;
144
    }
145
    mChillDaysAfter = days_after;
146
}