Loading [MathJax]/extensions/tex2jax.js
LSST Applications g04a91732dc+7fec47d7bc,g07dc498a13+5ab4d22ec3,g0fba68d861+565de8e5d5,g1409bbee79+5ab4d22ec3,g1a7e361dbc+5ab4d22ec3,g1fd858c14a+11200c7927,g20f46db602+25d63fd678,g35bb328faa+fcb1d3bbc8,g4d2262a081+61302e889d,g4d39ba7253+d05e267ece,g4e0f332c67+5d362be553,g53246c7159+fcb1d3bbc8,g60b5630c4e+d05e267ece,g78460c75b0+2f9a1b4bcd,g786e29fd12+cf7ec2a62a,g7b71ed6315+fcb1d3bbc8,g8048e755c2+a1301e4c20,g8852436030+163ceb82d8,g89139ef638+5ab4d22ec3,g89e1512fd8+fbb808ebce,g8d6b6b353c+d05e267ece,g9125e01d80+fcb1d3bbc8,g989de1cb63+5ab4d22ec3,g9f33ca652e+8abe617c77,ga9baa6287d+d05e267ece,gaaedd4e678+5ab4d22ec3,gabe3b4be73+1e0a283bba,gb1101e3267+fefe9ce5b1,gb58c049af0+f03b321e39,gb90eeb9370+824c420ec4,gc741bbaa4f+77ddc33078,gcf25f946ba+163ceb82d8,gd315a588df+0f88d5218e,gd6cbbdb0b4+c8606af20c,gd9a9a58781+fcb1d3bbc8,gde0f65d7ad+e6bd566e97,ge278dab8ac+932305ba37,ge82c20c137+76d20ab76d,w.2025.10
LSST Data Management Base Package
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
StorableMap.h
Go to the documentation of this file.
1// -*- LSST-C++ -*-
2/*
3 * This file is part of afw.
4 *
5 * Developed for the LSST Data Management System.
6 * This product includes software developed by the LSST Project
7 * (https://www.lsst.org).
8 * See the COPYRIGHT file at the top-level directory of this distribution
9 * for details of code ownership.
10 *
11 * This program is free software: you can redistribute it and/or modify
12 * it under the terms of the GNU General Public License as published by
13 * the Free Software Foundation, either version 3 of the License, or
14 * (at your option) any later version.
15 *
16 * This program is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19 * GNU General Public License for more details.
20 *
21 * You should have received a copy of the GNU General Public License
22 * along with this program. If not, see <https://www.gnu.org/licenses/>.
23 */
24
25#ifndef LSST_AFW_IMAGE_DETAIL_STORABLEMAP_H
26#define LSST_AFW_IMAGE_DETAIL_STORABLEMAP_H
27
28#include <exception>
29#include <memory>
30#include <sstream>
31#include <string>
32#include <type_traits>
33#include <unordered_map>
34#include <utility>
35
36#include "lsst/pex/exceptions.h"
39
40namespace lsst {
41namespace afw {
42namespace image {
43namespace detail {
44
59class StorableMap final {
60public:
69 using const_pointer = value_type const*;
70
71private:
73
74public:
75 // These definitions may be replaced with adapters if we need
76 // StorableMap's iterators to behave differently from _Impl's.
77 using const_iterator = _Impl::const_iterator;
78 using iterator = _Impl::iterator;
81
82 template <typename V>
85
87 StorableMap(StorableMap const& other);
91 ~StorableMap() noexcept;
92
105 StorableMap(std::initializer_list<value_type> init);
106
121 template <typename T>
122 std::shared_ptr<T> at(Key<std::shared_ptr<T>> const& key) const {
124 "Can only retrieve pointers to subclasses of Storable.");
125 static_assert(std::is_const<T>::value,
126 "Due to implementation constraints, pointers to non-const are not supported.");
127 try {
128 // unordered_map::at(Key<Storable>) does not do any type-checking.
129 mapped_type const& pointer = _contents.at(key);
130
131 // Null pointer stored; skip dynamic_cast because won't change result.
132 if (pointer == nullptr) {
133 return nullptr;
134 }
135
137 // shared_ptr can be empty without being null. dynamic_pointer_cast
138 // only promises result of failed cast is empty, so test for that.
139 if (typedPointer.use_count() > 0) {
140 return typedPointer;
141 } else {
142 std::stringstream message;
143 message << "Key " << key << " not found, but a key labeled " << key.getId() << " is present.";
145 }
146 } catch (std::out_of_range const&) {
148 "No key labeled " + key.getId() + " found."));
149 }
150 }
151
153 size_type size() const noexcept;
154
156 bool empty() const noexcept;
157
167 size_type max_size() const noexcept;
168
184 bool contains(std::string const& key) const;
185
200 template <typename T>
201 bool contains(Key<std::shared_ptr<T>> const& key) const {
203 "Can only retrieve pointers to subclasses of Storable.");
204 static_assert(std::is_const<T>::value,
205 "Due to implementation constraints, pointers to non-const are not supported.");
206 if (_contents.count(key) > 0) {
207 // unordered_map::at(Key<Storable>) does not do any type-checking.
208 mapped_type const& pointer = _contents.at(key);
209
210 // Null pointer stored; dynamic_cast will always return null.
211 if (pointer == nullptr) {
212 return true;
213 }
214
216 // shared_ptr can be empty without being null. dynamic_pointer_cast
217 // only promises result of failed cast is empty, so test for that.
218 return typedPointer.use_count() > 0;
219 } else {
220 return false;
221 }
222 }
223
232 bool operator==(StorableMap const& other) const noexcept;
233
234 bool operator!=(StorableMap const& other) const noexcept { return !(*this == other); }
235
237
243 void clear() noexcept;
244
262 template <typename T>
263 bool insert(Key<std::shared_ptr<T>> const& key, std::shared_ptr<T> const& value) {
265 "Can only store shared pointers to subclasses of Storable.");
266 static_assert(std::is_const<T>::value,
267 "Due to implementation constraints, pointers to non-const are not supported.");
268 // unordered_map uses Key<shared_ptr<Storable>> internally, so
269 // any key with the same ID will block emplacement.
270 return _contents.emplace(key, value).second;
271 }
272
292 template <typename T>
293 std::pair<Key<T>, bool> insert(std::string const& key, T const& value) {
294 auto strongKey = typehandling::makeKey<T>(key);
295 // Construct return value in advance, so that exception from
296 // copying/moving Key is atomic.
297 auto result = std::make_pair(strongKey, false);
298 result.second = insert(strongKey, value);
299 return result;
300 }
301
310 template <typename T>
311 bool erase(Key<T> const& key) noexcept {
312 // unordered_map::erase(Key<Storable>) does no type checking.
313 if (this->contains(key)) {
314 return _contents.erase(key) > 0;
315 } else {
316 return false;
317 }
318 }
319
328 iterator begin() noexcept;
329 const_iterator begin() const noexcept;
330 const_iterator cbegin() const noexcept { return begin(); }
331
333
342 iterator end() noexcept;
343 const_iterator end() const noexcept;
344 const_iterator cend() const noexcept { return end(); }
345
347
348private:
349 _Impl _contents;
350};
351
352} // namespace detail
353} // namespace image
354} // namespace afw
355} // namespace lsst
356
357#endif
#define LSST_EXCEPT(type,...)
Create an exception with a given type.
Definition Exception.h:48
std::reverse_iterator< iterator > reverse_iterator
Definition StorableMap.h:79
StorableMap & operator=(StorableMap &&other)
bool empty() const noexcept
Return true if this map contains no key-value pairs.
std::shared_ptr< T > at(Key< std::shared_ptr< T > > const &key) const
Return a the mapped value of the element with key equal to key.
bool contains(std::string const &key) const
Return true if this map contains a mapping whose key has the specified label.
lsst::afw::typehandling::Key< std::string, V > Key
Definition StorableMap.h:83
bool operator!=(StorableMap const &other) const noexcept
Test for map equality.
size_type size() const noexcept
Return the number of key-value pairs in the map.
typehandling::Key< std::string, mapped_type > key_type
Definition StorableMap.h:62
bool erase(Key< T > const &key) noexcept
Remove the mapping for a key from this map, if it exists.
const_iterator cend() const noexcept
Return an iterator to the element past the end of the map.
const_iterator cbegin() const noexcept
Return an iterator to the first element of the map.
iterator begin() noexcept
Return an iterator to the first element of the map.
bool insert(Key< std::shared_ptr< T > > const &key, std::shared_ptr< T > const &value)
Insert an element into the map, if the map doesn't already contain a mapping with the same or a confl...
StorableMap(StorableMap const &other)
_Impl::const_iterator const_iterator
Definition StorableMap.h:77
iterator end() noexcept
Return an iterator to the element past the end of the map.
StorableMap & operator=(StorableMap const &other)
std::shared_ptr< typehandling::Storable const > mapped_type
Definition StorableMap.h:61
lsst::afw::typehandling::Storable Storable
Definition StorableMap.h:84
std::pair< Key< T >, bool > insert(std::string const &key, T const &value)
Insert an element into the map, if the map doesn't already contain a mapping with a conflicting key.
std::pair< key_type const, mapped_type > value_type
Definition StorableMap.h:63
size_type max_size() const noexcept
Return the maximum number of elements the container is able to hold due to system or library implemen...
std::reverse_iterator< const_iterator > const_reverse_iterator
Definition StorableMap.h:80
Key for type-safe lookup in a GenericMap.
Definition Key.h:52
Interface supporting iteration over heterogenous containers.
Definition Storable.h:58
Reports attempts to access elements outside a valid range of indices.
Definition Runtime.h:89
T make_pair(T... args)
constexpr Key< K, V > makeKey(K const &id)
Factory function for Key, to enable type parameter inference.
Definition Key.h:173
STL namespace.
T dynamic_pointer_cast(T... args)
T str(T... args)
T throw_with_nested(T... args)