OOPS
FieldL95.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  * In applying this licence, ECMWF does not waive the privileges and immunities
7  * granted to it by virtue of its status as an intergovernmental organisation nor
8  * does it submit to any jurisdiction.
9  */
10 
11 #include "lorenz95/FieldL95.h"
12 
13 #include <cmath>
14 #include <fstream>
15 #include <limits>
16 #include <string>
17 
18 #include "eckit/config/Configuration.h"
19 #include "eckit/exception/Exceptions.h"
20 
21 #include "lorenz95/GomL95.h"
22 #include "lorenz95/LocsL95.h"
23 #include "lorenz95/Resolution.h"
24 #include "oops/util/abor1_cpp.h"
25 #include "oops/util/Logger.h"
26 #include "oops/util/Random.h"
27 
28 // -----------------------------------------------------------------------------
29 namespace lorenz95 {
30 // -----------------------------------------------------------------------------
32  : resol_(resol.npoints()), x_(resol_)
33 {
34  ASSERT(resol_ > 0);
35  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 0.0;
36 }
37 // -----------------------------------------------------------------------------
38 FieldL95::FieldL95(const FieldL95 & other, const Resolution & resol)
39  : resol_(resol.npoints()), x_(resol_)
40 {
41  ASSERT(resol_ > 0);
42  ASSERT(other.resol_ == resol_);
43  for (int jj = 0; jj < resol_; ++jj) x_[jj] = other.x_[jj];
44 }
45 // -----------------------------------------------------------------------------
46 FieldL95::FieldL95(const FieldL95 & other, const bool copy)
47  : resol_(other.resol_), x_(resol_)
48 {
49  ASSERT(resol_ > 0);
50  if (copy) {
51  for (int jj = 0; jj < resol_; ++jj) x_[jj] = other.x_[jj];
52  } else {
53  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 0.0;
54  }
55 }
56 // -----------------------------------------------------------------------------
58  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 0.0;
59 }
60 // -----------------------------------------------------------------------------
62  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 1.0;
63 }
64 // -----------------------------------------------------------------------------
65 void FieldL95::dirac(const eckit::Configuration & config) {
66 // Get Diracs position
67  std::vector<int> ixdir(config.getIntVector("ixdir"));
68 
69 // Check
70  ASSERT(ixdir.size() > 0);
71  for (unsigned int jj = 0; jj < ixdir.size(); ++jj) {
72  ASSERT(ixdir[jj] < resol_);
73  }
74 
75 // Setup Dirac
76  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 0.0;
77  for (unsigned int jj = 0; jj < ixdir.size(); ++jj) x_[ixdir[jj]] = 1.0;
78 }
79 // -----------------------------------------------------------------------------
80 void FieldL95::generate(const eckit::Configuration & conf) {
81  for (int jj = 0; jj < resol_; ++jj) x_[jj] = 0.0;
82  if (conf.has("mean")) {
83  const double zz = conf.getDouble("mean");
84  for (int jj = 0; jj < resol_; ++jj) x_[jj] = zz;
85  }
86  if (conf.has("sinus")) {
87  const double zz = conf.getDouble("sinus");
88  const double pi = std::acos(-1.0);
89  const double dx = 2.0 * pi / static_cast<double>(resol_);
90  for (int jj = 0; jj < resol_; ++jj) x_[jj] += zz * std::sin(static_cast<double>(jj) * dx);
91  }
92  if (conf.has("dirac")) {
93  const int ii = conf.getInt("dirac");
94  x_[ii] += 1.0;
95  }
96  oops::Log::trace() << "FieldL95::generate " << x_[28] << ", " << x_[29] << std::endl;
97 }
98 // -----------------------------------------------------------------------------
100  ASSERT(rhs.resol_ == resol_);
101  for (int jj = 0; jj < resol_; ++jj) x_[jj] = rhs.x_[jj];
102  return *this;
103 }
104 // -----------------------------------------------------------------------------
106  ASSERT(rhs.resol_ == resol_);
107  for (int jj = 0; jj < resol_; ++jj) x_[jj] += rhs.x_[jj];
108  return *this;
109 }
110 // -----------------------------------------------------------------------------
112  ASSERT(rhs.resol_ == resol_);
113  for (int jj = 0; jj < resol_; ++jj) x_[jj] -= rhs.x_[jj];
114  return *this;
115 }
116 // -----------------------------------------------------------------------------
117 FieldL95 & FieldL95::operator*= (const double & fact) {
118  for (int jj = 0; jj < resol_; ++jj) x_[jj] *= fact;
119  return *this;
120 }
121 // -----------------------------------------------------------------------------
122 void FieldL95::diff(const FieldL95 & x1, const FieldL95 & x2) {
123  ASSERT(x1.resol_ == resol_);
124  ASSERT(x2.resol_ == resol_);
125  for (int jj = 0; jj < resol_; ++jj) {
126  x_[jj] = x1.x_[jj] - x2.x_[jj];
127  }
128 }
129 // -----------------------------------------------------------------------------
130 void FieldL95::axpy(const double & zz, const FieldL95 & rhs) {
131  ASSERT(rhs.resol_ == resol_);
132  for (int jj = 0; jj < resol_; ++jj) x_[jj] += zz * rhs.x_[jj];
133 }
134 // -----------------------------------------------------------------------------
135 double FieldL95::dot_product_with(const FieldL95 & other) const {
136  ASSERT(other.resol_ == resol_);
137  double zz = 0.0;
138  for (int jj = 0; jj < resol_; ++jj) zz += x_[jj] * other.x_[jj];
139  return zz;
140 }
141 // -----------------------------------------------------------------------------
142 void FieldL95::schur(const FieldL95 & rhs) {
143  ASSERT(rhs.resol_ == resol_);
144  for (int jj = 0; jj < resol_; ++jj) x_[jj] *= rhs.x_[jj];
145 }
146 // -----------------------------------------------------------------------------
148  util::NormalDistribution<double> xx(resol_, 0.0, 1.0, 1);
149  for (int jj = 0; jj < resol_; ++jj) x_[jj] = xx[jj];
150 }
151 // -----------------------------------------------------------------------------
152 void FieldL95::read(std::ifstream & fin) {
153  fin.precision(std::numeric_limits<double>::digits10);
154  for (int jj = 0; jj < resol_; ++jj) fin >> x_[jj];
155 }
156 // -----------------------------------------------------------------------------
157 void FieldL95::write(std::ofstream & fout) const {
158  fout.precision(std::numeric_limits<double>::digits10);
159  for (int jj = 0; jj < resol_; ++jj) fout << x_[jj] << " ";
160 }
161 // -----------------------------------------------------------------------------
162 double FieldL95::rms() const {
163  double zz = 0.0;
164  for (int jj = 0; jj < resol_; ++jj) zz += x_[jj] * x_[jj];
165  zz = sqrt(zz/resol_);
166  return zz;
167 }
168 // -----------------------------------------------------------------------------
169 size_t FieldL95::serialSize() const {
170  return resol_;
171 }
172 // -----------------------------------------------------------------------------
173 void FieldL95::serialize(std::vector<double> & vect) const {
174  vect.insert(vect.end(), x_.begin(), x_.end());
175 }
176 // -----------------------------------------------------------------------------
177 void FieldL95::deserialize(const std::vector<double> & vect, size_t & index) {
178  for (int ii = 0; ii < resol_; ++ii) {
179  x_[ii] = vect[index];
180  ++index;
181  }
182 }
183 // -----------------------------------------------------------------------------
184 void FieldL95::print(std::ostream & os) const {
185  double zmin = x_[0];
186  double zmax = x_[0];
187  double zavg = 0.0;
188  for (int jj = 0; jj < resol_; ++jj) {
189  if (x_[jj] < zmin) zmin = x_[jj];
190  if (x_[jj] > zmax) zmax = x_[jj];
191  zavg += x_[jj];
192  }
193  zavg /= resol_;
194  os << " Min=" << zmin << ", Max=" << zmax << ", Average=" << zavg;
195 }
196 // -----------------------------------------------------------------------------
197 
198 } // namespace lorenz95
Class to represent fields for the L95 model.
Definition: FieldL95.h:34
FieldL95(const Resolution &)
Constructors and basic operators.
Definition: FieldL95.cc:31
void generate(const eckit::Configuration &)
Definition: FieldL95.cc:80
FieldL95 & operator-=(const FieldL95 &)
Definition: FieldL95.cc:111
FieldL95 & operator+=(const FieldL95 &)
Definition: FieldL95.cc:105
void write(std::ofstream &) const
Definition: FieldL95.cc:157
void diff(const FieldL95 &, const FieldL95 &)
Definition: FieldL95.cc:122
void deserialize(const std::vector< double > &, size_t &) override
Definition: FieldL95.cc:177
void axpy(const double &, const FieldL95 &)
Definition: FieldL95.cc:130
const int resol_
Definition: FieldL95.h:78
void schur(const FieldL95 &)
Definition: FieldL95.cc:142
void zero()
Linear algebra.
Definition: FieldL95.cc:57
FieldL95 & operator*=(const double &)
Definition: FieldL95.cc:117
FieldL95 & operator=(const FieldL95 &)
Definition: FieldL95.cc:99
void dirac(const eckit::Configuration &)
Definition: FieldL95.cc:65
double dot_product_with(const FieldL95 &) const
Definition: FieldL95.cc:135
std::vector< double > x_
Definition: FieldL95.h:79
void serialize(std::vector< double > &) const override
Definition: FieldL95.cc:173
void read(std::ifstream &)
Utilities.
Definition: FieldL95.cc:152
size_t serialSize() const override
Serialize and deserialize.
Definition: FieldL95.cc:169
void print(std::ostream &) const override
Definition: FieldL95.cc:184
double rms() const
Definition: FieldL95.cc:162
Handles resolution.
Definition: Resolution.h:43
The namespace for the L95 model.
real(kind_real), parameter, public pi
Pi.