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 |