OOPS
ObsVec1D.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  * (C) Copyright 2020-2020 UCAR.
4  *
5  * This software is licensed under the terms of the Apache Licence Version 2.0
6  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
7  * In applying this licence, ECMWF does not waive the privileges and immunities
8  * granted to it by virtue of its status as an intergovernmental organisation nor
9  * does it submit to any jurisdiction.
10  */
11 
12 #include "lorenz95/ObsVec1D.h"
13 
14 #include <math.h>
15 #include <limits>
16 
17 #include "eckit/config/Configuration.h"
18 #include "eckit/exception/Exceptions.h"
19 #include "lorenz95/ObsData1D.h"
20 #include "lorenz95/ObsTable.h"
21 #include "oops/util/Logger.h"
22 #include "oops/util/missingValues.h"
23 
24 namespace lorenz95 {
25 // -----------------------------------------------------------------------------
27  const std::string & name)
28  : obsdb_(ot), data_(ot.nobs()), missing_(util::missingValue(missing_))
29 {
30  for (double & val : data_) { val = 0.0; }
31  if (!name.empty()) obsdb_.getdb(name, data_);
32 }
33 // -----------------------------------------------------------------------------
35  : obsdb_(other.obsdb_), data_(other.data_.size()), missing_(util::missingValue(missing_))
36 {
37  data_ = other.data_;
38 }
39 // -----------------------------------------------------------------------------
41  ASSERT(data_.size() == rhs.data_.size());
42  data_ = rhs.data_;
43  return *this;
44 }
45 
46 // -----------------------------------------------------------------------------
47 ObsVec1D & ObsVec1D::operator*= (const double & zz) {
48  for (double & val : data_) {
49  if (val != missing_) val *= zz;
50  }
51  return *this;
52 }
53 // -----------------------------------------------------------------------------
55  ASSERT(data_.size() == rhs.data_.size());
56  for (size_t jj = 0; jj < data_.size(); ++jj) {
57  if (data_[jj] == missing_ || rhs.data_[jj] == missing_) {
58  data_[jj] = missing_;
59  } else {
60  data_[jj] += rhs.data_[jj];
61  }
62  }
63  return *this;
64 }
65 // -----------------------------------------------------------------------------
67  ASSERT(data_.size() == rhs.data_.size());
68  for (size_t jj = 0; jj < data_.size(); ++jj) {
69  if (data_[jj] == missing_ || rhs.data_[jj] == missing_) {
70  data_[jj] = missing_;
71  } else {
72  data_[jj] -= rhs.data_[jj];
73  }
74  }
75  return *this;
76 }
77 // -----------------------------------------------------------------------------
79  ASSERT(data_.size() == rhs.data_.size());
80  for (size_t jj = 0; jj < data_.size(); ++jj) {
81  if (data_[jj] == missing_ || rhs.data_[jj] == missing_) {
82  data_[jj] = missing_;
83  } else {
84  data_[jj] *= rhs.data_[jj];
85  }
86  }
87  return *this;
88 }
89 // -----------------------------------------------------------------------------
91  ASSERT(data_.size() == rhs.data_.size());
92  for (size_t jj = 0; jj < data_.size(); ++jj) {
93  if (data_[jj] == missing_ || rhs.data_[jj] == missing_) {
94  data_[jj] = missing_;
95  } else {
96  data_[jj] /= rhs.data_[jj];
97  }
98  }
99  return *this;
100 }
101 // -----------------------------------------------------------------------------
103  for (double & val : data_) val = 0.0;
104 }
105 // -----------------------------------------------------------------------------
107  for (double & val : data_) val = 1.0;
108 }
109 // -----------------------------------------------------------------------------
111  for (double & val : data_) {
112  if (val != missing_) val = 1.0/val;
113  }
114 }
115 // -----------------------------------------------------------------------------
116 void ObsVec1D::axpy(const double & zz, const ObsVec1D & rhs) {
117  ASSERT(data_.size() == rhs.data_.size());
118  for (size_t jj = 0; jj < data_.size(); ++jj) {
119  if (data_[jj] == missing_ || rhs.data_[jj] == missing_) {
120  data_[jj] = missing_;
121  } else {
122  data_[jj] += zz * rhs.data_[jj];
123  }
124  }
125 }
126 // -----------------------------------------------------------------------------
129 }
130 // -----------------------------------------------------------------------------
131 double ObsVec1D::dot_product_with(const ObsVec1D & other) const {
132  ASSERT(data_.size() == other.data_.size());
133  double zz = 0.0;
134  for (size_t jj = 0; jj < data_.size(); ++jj) {
135  if ((data_[jj] != missing_) && (other.data_[jj] != missing_)) {
136  zz += data_[jj] * other.data_[jj];
137  }
138  }
139  return zz;
140 }
141 // -----------------------------------------------------------------------------
142 double ObsVec1D::rms() const {
143  double zz = 0.0;
144  double iobs = 0.0;
145  for (const double & val : data_) {
146  if (val != missing_) {
147  zz += val * val;
148  iobs += 1.0;
149  }
150  }
151  if (iobs > 0.0) zz = sqrt(zz/iobs);
152  return zz;
153 }
154 // -----------------------------------------------------------------------------
155 unsigned int ObsVec1D::nobs() const {
156  return data_.size() - std::count(data_.begin(), data_.end(), missing_);
157 }
158 // -----------------------------------------------------------------------------
159 void ObsVec1D::mask(const ObsData1D<int> & mask) {
160  for (size_t jj = 0; jj < data_.size(); ++jj) {
161  if (mask[jj]) data_.at(jj) = missing_;
162  }
163 }
164 // -----------------------------------------------------------------------------
165 void ObsVec1D::mask(const ObsVec1D & mask) {
166  for (size_t jj = 0; jj < data_.size(); ++jj) {
167  if (mask[jj] == missing_) data_.at(jj) = missing_;
168  }
169 }
170 // -----------------------------------------------------------------------------
172  const float fmiss = util::missingValue(float());
173  for (size_t jj = 0; jj < data_.size(); ++jj) {
174  if (rhs[jj] == fmiss) {
175  data_.at(jj) = missing_;
176  } else {
177  data_.at(jj) = static_cast<double>(rhs[jj]);
178  }
179  }
180  return *this;
181 }
182 // -----------------------------------------------------------------------------
183 void ObsVec1D::save(const std::string & name) const {
184  obsdb_.putdb(name, data_);
185 }
186 // -----------------------------------------------------------------------------
187 Eigen::VectorXd ObsVec1D::packEigen(const ObsVec1D & mask) const {
188  Eigen::VectorXd vec(packEigenSize(mask));
189  size_t ii = 0;
190  for (size_t jj = 0; jj < data_.size(); ++jj) {
191  if ((data_[jj] != missing_) && (mask[jj] != missing_)) {
192  vec(ii++) = data_[jj];
193  }
194  }
195  return vec;
196 }
197 // -----------------------------------------------------------------------------
198 size_t ObsVec1D::packEigenSize(const ObsVec1D & mask) const {
199  size_t ii = 0;
200  for (size_t jj = 0; jj < data_.size(); ++jj) {
201  if ((data_[jj] != missing_) && (mask[jj] != missing_)) {
202  ii++;
203  }
204  }
205  return ii;
206 }
207 // -----------------------------------------------------------------------------
208 void ObsVec1D::read(const std::string & name) {
209  obsdb_.getdb(name, data_);
210 }
211 // -----------------------------------------------------------------------------
212 void ObsVec1D::print(std::ostream & os) const {
213  double zmin = std::numeric_limits<double>::max();
214  double zmax = std::numeric_limits<double>::lowest();
215  double zavg = 0.0;
216  size_t iobs = 0;
217  for (const double & val : data_) {
218  if (val != missing_) {
219  if (val < zmin) zmin = val;
220  if (val > zmax) zmax = val;
221  zavg += val;
222  ++iobs;
223  }
224  }
225  if (iobs > 0) {
226  zavg /= static_cast<double>(iobs);
227  os << "Lorenz 95 nobs= " << iobs << " Min=" << zmin << ", Max=" << zmax
228  << ", Average=" << zavg;
229  } else {
230  os << "Lorenz 95 : No observations";
231  }
232 }
233 // -----------------------------------------------------------------------------
234 } // namespace lorenz95
Data in observation space.
Definition: ObsData1D.h:35
A Simple Observation Data Handler.
Definition: ObsTable.h:67
void random(std::vector< double > &) const
Definition: ObsTable.cc:228
void getdb(const std::string &, std::vector< int > &) const
Definition: ObsTable.cc:128
void putdb(const std::string &, const std::vector< int > &) const
Definition: ObsTable.cc:84
Vector in observation space.
Definition: ObsVec1D.h:33
unsigned int nobs() const
Definition: ObsVec1D.cc:155
void mask(const ObsData1D< int > &)
Definition: ObsVec1D.cc:159
std::vector< double > data_
Definition: ObsVec1D.h:81
const ObsTable & obsdb_
Definition: ObsVec1D.h:80
size_t packEigenSize(const ObsVec1D &) const
Definition: ObsVec1D.cc:198
void axpy(const double &, const ObsVec1D &)
Definition: ObsVec1D.cc:116
double rms() const
Definition: ObsVec1D.cc:142
void ones()
set all values to ones (for tests)
Definition: ObsVec1D.cc:106
Eigen::VectorXd packEigen(const ObsVec1D &) const
Definition: ObsVec1D.cc:187
ObsVec1D & operator=(const ObsVec1D &)
Definition: ObsVec1D.cc:40
ObsVec1D & operator*=(const double &)
Definition: ObsVec1D.cc:47
const double missing_
Definition: ObsVec1D.h:82
void print(std::ostream &) const
Definition: ObsVec1D.cc:212
double dot_product_with(const ObsVec1D &) const
Definition: ObsVec1D.cc:131
void read(const std::string &)
Definition: ObsVec1D.cc:208
ObsVec1D & operator-=(const ObsVec1D &)
Definition: ObsVec1D.cc:66
ObsVec1D & operator/=(const ObsVec1D &)
Definition: ObsVec1D.cc:90
void save(const std::string &) const
Definition: ObsVec1D.cc:183
ObsVec1D & operator+=(const ObsVec1D &)
Definition: ObsVec1D.cc:54
ObsVec1D(const ObsTable &, const std::string &name="")
Definition: ObsVec1D.cc:26
The namespace for the L95 model.
Definition: TLML95.h:34