opm-common
Loading...
Searching...
No Matches
SparseVector.hpp
1//===========================================================================
2//
3// File: SparseVector.hpp
4//
5// Created: Mon Jun 29 15:28:59 2009
6//
7// Author(s): Atgeirr F Rasmussen <atgeirr@sintef.no>
8// Bård Skaflestad <bard.skaflestad@sintef.no>
9//
10// $Date$
11//
12// $Revision$
13//
14//===========================================================================
15
16/*
17 Copyright 2009, 2010 SINTEF ICT, Applied Mathematics.
18 Copyright 2009, 2010 Statoil ASA.
19
20 This file is part of the Open Porous Media project (OPM).
21
22 OPM is free software: you can redistribute it and/or modify
23 it under the terms of the GNU General Public License as published by
24 the Free Software Foundation, either version 3 of the License, or
25 (at your option) any later version.
26
27 OPM is distributed in the hope that it will be useful,
28 but WITHOUT ANY WARRANTY; without even the implied warranty of
29 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
30 GNU General Public License for more details.
31
32 You should have received a copy of the GNU General Public License
33 along with OPM. If not, see <http://www.gnu.org/licenses/>.
34*/
35
36
37#ifndef OPM_SPARSEVECTOR_HEADER
38#define OPM_SPARSEVECTOR_HEADER
39
40#include <vector>
41#include <numeric>
42#include <algorithm>
43#include <opm/common/ErrorMacros.hpp>
44
45namespace Opm
46{
47
53 template <typename T>
55 {
56 public:
59 : size_(0), default_elem_()
60 {
61 }
62
65 explicit SparseVector(int sz)
66 : size_(sz), default_elem_()
67 {
68 }
69
76 template <typename DataIter, typename IntegerIter>
78 DataIter data_beg, DataIter data_end,
79 IntegerIter index_beg, IntegerIter index_end)
80 : size_(sz), data_(data_beg, data_end), indices_(index_beg, index_end),
81 default_elem_()
82 {
83#ifndef NDEBUG
84 OPM_ERROR_IF(sz < 0, "The size of a SparseVector must be non-negative");
85 OPM_ERROR_IF(indices_.size() != data_.size(), "The number of indices of a SparseVector must equal to the number of entries");
86 int last_index = -1;
87 int num_ind = indices_.size();
88 for (int i = 0; i < num_ind; ++i) {
89 int index = indices_[i];
90 if (index <= last_index || index >= sz) {
91 OPM_THROW(std::logic_error, "Error in SparseVector construction, index is nonincreasing or out of range.");
92 }
93 last_index = index;
94 }
95#endif
96 }
97
98
102 void addElement(const T& elem, int index)
103 {
104 assert(indices_.empty() || index > indices_.back());
105 assert(index < size_);
106 data_.push_back(elem);
107 indices_.push_back(index);
108 }
109
111 bool empty() const
112 {
113 return size_ == 0;
114 }
115
118 int size() const
119 {
120 return size_;
121 }
122
124 int nonzeroSize() const
125 {
126 return data_.size();
127 }
128
130 void clear()
131 {
132 data_.clear();
133 indices_.clear();
134 size_ = 0;
135 }
136
138 bool operator==(const SparseVector& other) const
139 {
140 return size_ == other.size_ && data_ == other.data_ && indices_ == other.indices_;
141 }
142
147 const T& element(int index) const
148 {
149#ifndef NDEBUG
150 OPM_ERROR_IF(index < 0,
151 "The index of a SparseVector must be non-negative "
152 "(is " + std::to_string(index) + ")");
153 OPM_ERROR_IF(index >= size_,
154 "The index of a SparseVector must be smaller "
155 "than the maximum value "
156 "(is " + std::to_string(index) +
157 ", max value: " + std::to_string(size_) + ")");
158#endif
159 std::vector<int>::const_iterator lb = std::lower_bound(indices_.begin(), indices_.end(), index);
160 if (lb != indices_.end() && *lb == index) {
161 return data_[lb - indices_.begin()];
162 } else {
163 return default_elem_;
164 }
165 }
166
170 const T& nonzeroElement(int nzindex) const
171 {
172#ifndef NDEBUG
173 OPM_ERROR_IF(nzindex < 0,
174 "The index of a SparseVector must be non-negative "
175 "(is " + std::to_string(nzindex) + ")");
176 OPM_ERROR_IF(nzindex >= nonzeroSize(),
177 "The index of a SparseVector must be smaller "
178 "than the maximum value " +
179 "(is " + std::to_string(nzindex) +
180 ", max value: " + std::to_string(nonzeroSize()) + ")");
181#endif
182 return data_[nzindex];
183 }
184
188 int nonzeroIndex(int nzindex) const
189 {
190 assert(nzindex >= 0);
191 assert(nzindex < nonzeroSize());
192 return indices_[nzindex];
193 }
194
195 private:
196 // The vectors data_ and indices_ are always the same size.
197 // The indices are supposed to be stored in increasing order,
198 // to be unique, and to be in [0, size_ - 1].
199 // default_elem_ is returned when a default element is requested.
200 int size_;
201 std::vector<T> data_;
202 std::vector<int> indices_;
203 T default_elem_;
204 };
205
206} // namespace Opm
207
208
209
210
211#endif // OPM_SPARSEVECTOR_HEADER
int size() const
Returns the size of the vector.
Definition SparseVector.hpp:118
void clear()
Makes the vector empty().
Definition SparseVector.hpp:130
void addElement(const T &elem, int index)
Appends an element to the vector.
Definition SparseVector.hpp:102
int nonzeroIndex(int nzindex) const
O(1) index access.
Definition SparseVector.hpp:188
SparseVector(int sz, DataIter data_beg, DataIter data_end, IntegerIter index_beg, IntegerIter index_end)
A constructor taking all the element data for the vector and their indices.
Definition SparseVector.hpp:77
int nonzeroSize() const
Returns the number of nonzero data elements.
Definition SparseVector.hpp:124
bool empty() const
Definition SparseVector.hpp:111
const T & nonzeroElement(int nzindex) const
O(1) element access.
Definition SparseVector.hpp:170
const T & element(int index) const
O(log n) element access.
Definition SparseVector.hpp:147
SparseVector()
Default constructor. Yields an empty SparseVector.
Definition SparseVector.hpp:58
bool operator==(const SparseVector &other) const
Equality.
Definition SparseVector.hpp:138
SparseVector(int sz)
Constructs a SparseVector with a given size, but no nonzero elements.
Definition SparseVector.hpp:65
This class implements a small container which holds the transmissibility mulitpliers for all the face...
Definition Exceptions.hpp:30