forked from perabrahamsen/daisy-model
-
Notifications
You must be signed in to change notification settings - Fork 0
/
adsorption_langmuir.C
122 lines (111 loc) · 3.92 KB
/
adsorption_langmuir.C
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
// adsorption_langmuir.C
//
// Copyright 1996-2001 Per Abrahamsen and Søren Hansen
// Copyright 2000-2001 KVL.
//
// This file is part of Daisy.
//
// Daisy is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser Public License as published by
// the Free Software Foundation; either version 2.1 of the License, or
// (at your option) any later version.
//
// Daisy is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser Public License for more details.
//
// You should have received a copy of the GNU Lesser Public License
// along with Daisy; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
#define BUILD_DLL
#include "adsorption.h"
#include "block_model.h"
#include "soil.h"
#include "check.h"
#include "mathlib.h"
#include "librarian.h"
#include "treelog.h"
#include "frame.h"
class AdsorptionLangmuir : public Adsorption
{
// Parameters.
const double K;
const double my_max_clay;
const double my_max_OC;
// Simulation.
public:
double C_to_M (const Soil& soil, const double Theta,
const int i, const double C, double sf) const
{
const double my_max
= my_max_clay * soil.clay (i) + my_max_OC * soil.humus (i);
const double S = (my_max * C) / (K + C);
return sf * soil.dry_bulk_density (i) * S + Theta * C;
}
double M_to_C (const Soil& soil, const double Theta,
const int i, const double M, double sf) const
{
// We need to solve the following equation w.r.t. C.
//
// M = rho (my_max C) / (K + C) + Theta C
// ==>
// M (K + C) = rho my_max C + Theta C (K + C)
// ==>
// 0 = Theta C^2 + (rho my_max + Theta K - M) C - M K
//
// So we get a square equation. We use the positive solution.
const double my_max = my_max_clay * soil.clay (i);
const double a = Theta;
const double b = sf * soil.dry_bulk_density (i) * my_max + Theta * K - M;
const double c = - M * K;
return single_positive_root_of_square_equation (a, b, c);
}
// Create.
public:
AdsorptionLangmuir (const BlockModel& al)
: Adsorption (al),
K (al.number ("K")),
my_max_clay (al.number ("my_max_clay", 0.0)),
my_max_OC (al.check ("my_max_OC")
? al.number ("my_max_OC")
: al.number ("my_max_clay"))
{ }
};
static struct AdsorptionLangmuirSyntax : DeclareModel
{
Model* make (const BlockModel& al) const
{ return new AdsorptionLangmuir (al); }
static bool check_alist (const Metalib&, const Frame& al, Treelog& err)
{
bool ok = true;
const bool has_my_max_clay = al.check ("my_max_clay");
const bool has_my_max_OC = al.check ("my_max_OC");
if (!has_my_max_clay && !has_my_max_OC)
{
err.entry ("You must specify either 'my_max_clay' or 'my_max_OC'");
ok = false;
}
return ok;
}
AdsorptionLangmuirSyntax ()
: DeclareModel (Adsorption::component, "Langmuir", "\
M = rho (my_max C) / (K + C) + Theta C")
{ }
void load_frame (Frame& frame) const
{
frame.add_check (check_alist);
frame.declare ("K", "g/cm^3", Check::non_negative (), Attribute::Const, "Half saturation constant.");
frame.declare ("my_max_clay", "g/cm^3", Check::non_negative (),
Attribute::OptionalConst,
"Max adsorption capacity (clay).\n\
It is multiplied with the soil clay fraction to get the clay part of\n\
'my_max'. If 'my_max_OC' is specified, 'my_max_clay' defaults to 0.");
frame.declare ("my_max_OC", "g/cm^3", Check::non_negative (),
Attribute::OptionalConst,
"Max adsorption capacity (humus).\n\
It is multiplied with the soil organic carbon fraction to get the\n\
carbon part of 'my_max'. By default, 'my_max_OC' is equal to 'my_max_clay'.");
}
} AdsorptionLangmuir_syntax;
// adsorption_langmuir.C ends here.