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
 
587 werner 20
#include "carbonout.h"
21
#include "model.h"
22
#include "resourceunit.h"
23
#include "snag.h"
24
#include "soil.h"
25
 
26
CarbonOut::CarbonOut()
27
{
28
    setName("Carbon and nitrogen pools above and belowground per RU/yr", "carbon");
1157 werner 29
    setDescription("Carbon and nitrogen pools (C and N) per resource unit / year and/or by landsacpe/year. "\
30
                   "On resource unit level, the outputs contain aggregated above ground pools (kg/ha) " \
31
                   "and below ground pools (kg/ha). \n " \
32
                   "For landscape level outputs, all variables are scaled to kg/ha stockable area. "\
33
                   "The area column contains the stockable area (per resource unit / landscape) and can be used to scale to values to the actual value. \n " \
34
                   "You can use the 'condition' to control if the output should be created for the current year(see also dynamic stand output).\n" \
35
                   "The 'conditionRU' can be used to suppress resource-unit-level details; eg. specifying 'in(year,100,200,300)' limits output on reosurce unit level to the years 100,200,300 " \
36
                   "(leaving 'conditionRU' blank enables details per default).");
587 werner 37
    columns() << OutputColumn::year() << OutputColumn::ru() << OutputColumn::id()
1157 werner 38
              << OutputColumn("area_ha", "total stockable area of the resource unit (ha)", OutDouble)
39
              << OutputColumn("stem_c", "Stem carbon kg/ha", OutDouble)
40
              << OutputColumn("stem_n", "Stem nitrogen kg/ha", OutDouble)
41
              << OutputColumn("branch_c", "branches carbon kg/ha", OutDouble)
42
              << OutputColumn("branch_n", "branches nitrogen kg/ha", OutDouble)
43
              << OutputColumn("foliage_c", "Foliage carbon kg/ha", OutDouble)
44
              << OutputColumn("foliage_n", "Foliage nitrogen kg/ha", OutDouble)
45
              << OutputColumn("coarseRoot_c", "coarse root carbon kg/ha", OutDouble)
46
              << OutputColumn("coarseRoot_n", "coarse root nitrogen kg/ha", OutDouble)
47
              << OutputColumn("fineRoot_c", "fine root carbon kg/ha", OutDouble)
48
              << OutputColumn("fineRoot_n", "fine root nitrogen kg/ha", OutDouble)
49
              << OutputColumn("regeneration_c", "total carbon in regeneration layer (h<4m) kg/ha", OutDouble)
50
              << OutputColumn("regeneration_n", "total nitrogen in regeneration layer (h<4m) kg/ha", OutDouble)
51
              << OutputColumn("snags_c", "standing dead wood carbon kg/ha", OutDouble)
52
              << OutputColumn("snags_n", "standing dead wood nitrogen kg/ha", OutDouble)
53
              << OutputColumn("snagsOther_c", "branches and coarse roots of standing dead trees, carbon kg/ha", OutDouble)
54
              << OutputColumn("snagsOther_n", "branches and coarse roots of standing dead trees, nitrogen kg/ha", OutDouble)
592 werner 55
              << OutputColumn("downedWood_c", "downed woody debris (yR), carbon kg/ha", OutDouble)
56
              << OutputColumn("downedWood_n", "downed woody debris (yR), nitrogen kg/ga", OutDouble)
57
              << OutputColumn("litter_c", "soil litter (yl), carbon kg/ha", OutDouble)
58
              << OutputColumn("litter_n", "soil litter (yl), nitrogen kg/ha", OutDouble)
59
              << OutputColumn("soil_c", "soil organic matter (som), carbon kg/ha", OutDouble)
60
              << OutputColumn("soil_n", "soil organic matter (som), nitrogen kg/ha", OutDouble);
587 werner 61
 
62
 
63
}
64
 
65
void CarbonOut::setup()
66
{
837 werner 67
    // use a condition for to control execuation for the current year
68
    QString condition = settings().value(".condition", "");
69
    mCondition.setExpression(condition);
1157 werner 70
 
71
    condition = settings().value(".conditionRU", "");
72
    mConditionDetails.setExpression(condition);
73
 
587 werner 74
}
75
 
76
 
77
void CarbonOut::exec()
78
{
79
    Model *m = GlobalSettings::instance()->model();
837 werner 80
 
1157 werner 81
    // global condition
82
    if (!mCondition.isEmpty() && mCondition.calculate(GlobalSettings::instance()->currentYear())==0.)
83
        return;
84
 
85
    bool ru_level = true;
86
    // switch off details if this is indicated in the conditionRU option
87
    if (!mConditionDetails.isEmpty() && mConditionDetails.calculate(GlobalSettings::instance()->currentYear())==0.)
88
        ru_level = false;
89
 
90
 
91
    QVector<double> v(23, 0.); // 8 data values
92
    QVector<double>::iterator vit;
93
 
587 werner 94
    foreach(ResourceUnit *ru, m->ruList()) {
734 werner 95
        if (ru->id()==-1 || !ru->snag())
587 werner 96
            continue; // do not include if out of project area
1157 werner 97
 
587 werner 98
        const StandStatistics &s = ru->statistics();
1157 werner 99
        int ru_count = 0;
100
        double area_factor = ru->stockableArea() / cRUArea; // conversion factor from real area to per ha values
101
        if (ru_level) {
102
            *this << currentYear() << ru->index() << ru->id() << area_factor; // keys
103
            // biomass from trees (scaled to 1ha already)
587 werner 104
 
1157 werner 105
            *this << s.cStem() << s.nStem()   // stem
106
                               << s.cBranch() << s.nBranch()   // branch
107
                               << s.cFoliage() << s.nFoliage()   // foliage
108
                               << s.cCoarseRoot() << s.nCoarseRoot()   // coarse roots
109
                               << s.cFineRoot() << s.nFineRoot();   // fine roots
587 werner 110
 
1157 werner 111
            // biomass from regeneration
112
            *this << s.cRegeneration() << s.nRegeneration();
587 werner 113
 
1157 werner 114
            // biomass from standing dead wood
115
            *this << ru->snag()->totalSWD().C / area_factor << ru->snag()->totalSWD().N / area_factor   // snags
116
                                              << ru->snag()->totalOtherWood().C/area_factor << ru->snag()->totalOtherWood().N / area_factor;   // snags, other (branch + coarse root)
587 werner 117
 
1157 werner 118
            // biomass from soil (convert from t/ha -> kg/ha)
119
            *this << ru->soil()->youngRefractory().C*1000. << ru->soil()->youngRefractory().N*1000.   // wood
120
                                                           << ru->soil()->youngLabile().C*1000. << ru->soil()->youngLabile().N*1000.   // litter
121
                                                           << ru->soil()->oldOrganicMatter().C*1000. << ru->soil()->oldOrganicMatter().N*1000.;   // soil
122
 
123
            writeRow();
124
        }
125
        // landscape level statistics
126
 
127
        ++ru_count;
128
        vit = v.begin();
129
        *vit++ += area_factor;
130
        // carbon pools aboveground are in kg/resource unit, e.g., the sum of stem-carbon of all trees, so no scaling required
131
        *vit++ += s.cStem() * area_factor; *vit++ += s.nStem()* area_factor;
132
        *vit++ += s.cBranch() * area_factor; *vit++ += s.nBranch()* area_factor;
133
        *vit++ += s.cFoliage() * area_factor; *vit++ += s.nFoliage()* area_factor;
134
        *vit++ += s.cCoarseRoot() * area_factor; *vit++ += s.nCoarseRoot()* area_factor;
135
        *vit++ += s.cFineRoot() * area_factor; *vit++ += s.nFineRoot()* area_factor;
136
        // regen
137
        *vit++ += s.cRegeneration(); *vit++ += s.nRegeneration();
138
        // standing dead wood
139
        *vit++ += ru->snag()->totalSWD().C ; *vit++ += ru->snag()->totalSWD().N ;
140
        *vit++ += ru->snag()->totalOtherWood().C ; *vit++ += ru->snag()->totalOtherWood().N ;
141
        // biomass from soil (converstion to kg/ha), and scale with fraction of stockable area
142
        *vit++ += ru->soil()->youngRefractory().C*area_factor * 1000.; *vit++ += ru->soil()->youngRefractory().N*area_factor  * 1000.;
143
        *vit++ += ru->soil()->youngLabile().C*area_factor * 1000.; *vit++ += ru->soil()->youngLabile().N*area_factor  * 1000.;
144
        *vit++ += ru->soil()->oldOrganicMatter().C*area_factor  * 1000.; *vit++ += ru->soil()->oldOrganicMatter().N*area_factor  * 1000.;
145
 
587 werner 146
    }
1157 werner 147
    // write landscape sums
148
    double total_stockable_area = v[0]; // convert to ha of stockable area
149
    *this << currentYear() << -1 << -1; // keys
150
    *this << v[0]; // stockable area [m2]
151
    for (int i=1;i<v.size();++i)
152
        *this << v[i] / total_stockable_area;
153
    writeRow();
587 werner 154
 
155
}
156