OOPS
DualVector.h
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 #ifndef OOPS_ASSIMILATION_DUALVECTOR_H_
12 #define OOPS_ASSIMILATION_DUALVECTOR_H_
13 
14 #include <memory>
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
20 #include "oops/base/Departures.h"
22 #include "oops/base/Increment.h"
23 #include "oops/util/dot_product.h"
24 
25 namespace oops {
26 
27 /// Container of dual space vectors for all terms of the cost function.
28 /*!
29  * Contains dual space vectors for all terms of the cost function,
30  * that is Departures for Jo, an Increment_ for Jc, a ControlIncrement
31  * for Jb and Jq.
32  */
33 
34 // -----------------------------------------------------------------------------
35 template<typename MODEL, typename OBS> class DualVector {
39 
40  public:
41  DualVector(): dxjb_(), dxjo_(), dxjc_(), ijo_(), ijc_(), size_(0) {}
42  DualVector(const DualVector &);
44 
45 // Access increment
46  void dx(CtrlInc_ * dx) {dxjb_.reset(dx);}
47  const CtrlInc_ & dx() const {return *dxjb_;}
48  CtrlInc_ & dx() {return *dxjb_;}
49 
50 // Store and retrieve other elements (takes ownership)
51  void append(std::unique_ptr<GeneralizedDepartures> &&);
52  std::shared_ptr<const GeneralizedDepartures> getv(const unsigned) const;
53 
54 // Linear algebra
55  DualVector & operator=(const DualVector &);
56  DualVector & operator+=(const DualVector &);
57  DualVector & operator-=(const DualVector &);
58  DualVector & operator*=(const double);
59  void zero();
60  void axpy(const double, const DualVector &);
61  double dot_product_with(const DualVector &) const;
62 
63  void saveDep(const std::string &) const;
64 
65 // Clear everything
66  void clear();
67 
68  private:
69  bool compatible(const DualVector & other) const;
70 
71  std::unique_ptr<CtrlInc_> dxjb_;
72  std::vector<std::shared_ptr<Departures_> > dxjo_;
73  std::vector<std::shared_ptr<Increment_> > dxjc_;
74  std::vector<unsigned> ijo_;
75  std::vector<unsigned> ijc_;
76  unsigned size_;
77 };
78 
79 // =============================================================================
80 
81 template<typename MODEL, typename OBS>
83  : dxjb_(), dxjo_(), dxjc_(),
84  ijo_(other.ijo_), ijc_(other.ijc_), size_(other.size_)
85 {
86  if (other.dxjb_ != 0) {
87  dxjb_.reset(new CtrlInc_(*other.dxjb_));
88  }
89  for (unsigned jj = 0; jj < other.dxjo_.size(); ++jj) {
90  std::shared_ptr<Departures_> pd(new Departures_(*other.dxjo_[jj]));
91  dxjo_.push_back(pd);
92  }
93  for (unsigned jj = 0; jj < other.dxjc_.size(); ++jj) {
94  std::shared_ptr<Increment_> pi(new Increment_(*other.dxjc_[jj]));
95  dxjc_.push_back(pi);
96  }
97 }
98 // -----------------------------------------------------------------------------
99 template<typename MODEL, typename OBS>
101  dxjb_.reset();
102  dxjo_.clear();
103  dxjc_.clear();
104  ijo_.clear();
105  ijc_.clear();
106  size_ = 0;
107 }
108 // -----------------------------------------------------------------------------
109 template<typename MODEL, typename OBS>
110 void DualVector<MODEL, OBS>::append(std::unique_ptr<GeneralizedDepartures> && uv) {
111 // Since there is no duck-typing in C++, we do it manually.
112  std::shared_ptr<GeneralizedDepartures> sv = std::move(uv);
113  std::shared_ptr<Increment_> si = std::dynamic_pointer_cast<Increment_>(sv);
114  if (si != nullptr) {
115  dxjc_.push_back(si);
116  ijc_.push_back(size_);
117  }
118  std::shared_ptr<Departures_> sd = std::dynamic_pointer_cast<Departures_>(sv);
119  if (sd != nullptr) {
120  dxjo_.push_back(sd);
121  ijo_.push_back(size_);
122  }
123  ASSERT(si != nullptr || sd != nullptr);
124  ++size_;
125 }
126 // -----------------------------------------------------------------------------
127 template<typename MODEL, typename OBS>
128 std::shared_ptr<const GeneralizedDepartures>
129 DualVector<MODEL, OBS>::getv(const unsigned ii) const {
130  ASSERT(ii < size_);
131  std::shared_ptr<const GeneralizedDepartures> pv;
132  for (unsigned jj = 0; jj < ijo_.size(); ++jj) {
133  if (ijo_[jj] == ii) pv = dxjo_[jj];
134  }
135  for (unsigned jj = 0; jj < ijc_.size(); ++jj) {
136  if (ijc_[jj] == ii) pv = dxjc_[jj];
137  }
138  ASSERT(pv != 0);
139  return pv;
140 }
141 // -----------------------------------------------------------------------------
142 template<typename MODEL, typename OBS>
144  ASSERT(this->compatible(rhs));
145  if (dxjb_ != 0) {
146  *dxjb_ = *rhs.dxjb_;
147  }
148  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
149  *dxjo_[jj] = *rhs.dxjo_[jj];
150  }
151  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
152  *dxjc_[jj] = *rhs.dxjc_[jj];
153  }
154  return *this;
155 }
156 // -----------------------------------------------------------------------------
157 template<typename MODEL, typename OBS>
159  ASSERT(this->compatible(rhs));
160  if (dxjb_ != 0) {
161  *dxjb_ += *rhs.dxjb_;
162  }
163  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
164  *dxjo_[jj] += *rhs.dxjo_[jj];
165  }
166  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
167  *dxjc_[jj] += *rhs.dxjc_[jj];
168  }
169  return *this;
170 }
171 // -----------------------------------------------------------------------------
172 template<typename MODEL, typename OBS>
174  ASSERT(this->compatible(rhs));
175  if (dxjb_ != 0) {
176  *dxjb_ -= *rhs.dxjb_;
177  }
178  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
179  *dxjo_[jj] -= *rhs.dxjo_[jj];
180  }
181  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
182  *dxjc_[jj] -= *rhs.dxjc_[jj];
183  }
184  return *this;
185 }
186 // -----------------------------------------------------------------------------
187 template<typename MODEL, typename OBS>
189  if (dxjb_ != 0) {
190  *dxjb_ *= zz;
191  }
192  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
193  *dxjo_[jj] *= zz;
194  }
195  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
196  *dxjc_[jj] *= zz;
197  }
198  return *this;
199 }
200 // -----------------------------------------------------------------------------
201 template<typename MODEL, typename OBS>
203  if (dxjb_ != 0) {
204  dxjb_->zero();
205  }
206  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
207  dxjo_[jj]->zero();
208  }
209  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
210  dxjc_[jj]->zero();
211  }
212 }
213 // -----------------------------------------------------------------------------
214 template<typename MODEL, typename OBS>
215 void DualVector<MODEL, OBS>::axpy(const double zz, const DualVector & rhs) {
216  ASSERT(this->compatible(rhs));
217  if (dxjb_ != 0) {
218  dxjb_->axpy(zz, *rhs.dxjb_);
219  }
220  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
221  dxjo_[jj]->axpy(zz, *rhs.dxjo_[jj]);
222  }
223  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
224  dxjc_[jj]->axpy(zz, *rhs.dxjc_[jj]);
225  }
226 }
227 // -----------------------------------------------------------------------------
228 template<typename MODEL, typename OBS>
230  ASSERT(this->compatible(x2));
231  double zz = 0.0;
232  if (dxjb_ != 0) {
233  zz += dot_product(*dxjb_, *x2.dxjb_);
234  }
235  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
236  zz += dot_product(*dxjo_[jj], *x2.dxjo_[jj]);
237  }
238  for (unsigned jj = 0; jj < dxjc_.size(); ++jj) {
239  zz += dot_product(*dxjc_[jj], *x2.dxjc_[jj]);
240  }
241  return zz;
242 }
243 // -----------------------------------------------------------------------------
244 template<typename MODEL, typename OBS>
246  bool lcheck = (dxjb_ == 0) == (other.dxjb_ == 0)
247  && (dxjo_.size() == other.dxjo_.size())
248  && (dxjc_.size() == other.dxjc_.size());
249  return lcheck;
250 }
251 // -----------------------------------------------------------------------------
252 template<typename MODEL, typename OBS>
253 void DualVector<MODEL, OBS>::saveDep(const std::string & name) const {
254  for (unsigned jj = 0; jj < dxjo_.size(); ++jj) {
255  dxjo_[jj]->save(name);
256  }
257 }
258 // -----------------------------------------------------------------------------
259 } // namespace oops
260 
261 #endif // OOPS_ASSIMILATION_DUALVECTOR_H_
Difference between two observation vectors.
Definition: Departures.h:44
Container of dual space vectors for all terms of the cost function.
Definition: DualVector.h:35
ControlIncrement< MODEL, OBS > CtrlInc_
Definition: DualVector.h:37
std::vector< unsigned > ijc_
Definition: DualVector.h:75
std::vector< std::shared_ptr< Departures_ > > dxjo_
Definition: DualVector.h:72
void dx(CtrlInc_ *dx)
Definition: DualVector.h:46
const CtrlInc_ & dx() const
Definition: DualVector.h:47
bool compatible(const DualVector &other) const
Definition: DualVector.h:245
DualVector & operator*=(const double)
Definition: DualVector.h:188
CtrlInc_ & dx()
Definition: DualVector.h:48
unsigned size_
Definition: DualVector.h:76
DualVector & operator=(const DualVector &)
Definition: DualVector.h:143
Increment< MODEL > Increment_
Definition: DualVector.h:36
DualVector & operator+=(const DualVector &)
Definition: DualVector.h:158
double dot_product_with(const DualVector &) const
Definition: DualVector.h:229
void append(std::unique_ptr< GeneralizedDepartures > &&)
Definition: DualVector.h:110
Departures< OBS > Departures_
Definition: DualVector.h:38
std::shared_ptr< const GeneralizedDepartures > getv(const unsigned) const
Definition: DualVector.h:129
std::vector< std::shared_ptr< Increment_ > > dxjc_
Definition: DualVector.h:73
void axpy(const double, const DualVector &)
Definition: DualVector.h:215
std::unique_ptr< CtrlInc_ > dxjb_
Definition: DualVector.h:71
std::vector< unsigned > ijo_
Definition: DualVector.h:74
DualVector & operator-=(const DualVector &)
Definition: DualVector.h:173
void saveDep(const std::string &) const
Definition: DualVector.h:253
Increment class used in oops.
The namespace for the main oops code.
real(kind_real), parameter, public pi
Pi.