1 ///////////////////////////////////////////////////////////////////////////////
2 // MuldeR's Utilities for Qt
3 // Copyright (C) 2004-2023 LoRd_MuldeR <MuldeR2@GMX.de>
5 // This library is free software; you can redistribute it and/or
6 // modify it under the terms of the GNU Lesser General Public
7 // License as published by the Free Software Foundation; either
8 // version 2.1 of the License, or (at your option) any later version.
10 // This library is distributed in the hope that it will be useful,
11 // but WITHOUT ANY WARRANTY; without even the implied warranty of
12 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 // Lesser General Public License for more details.
15 // You should have received a copy of the GNU Lesser General Public
16 // License along with this library; if not, write to the Free Software
17 // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19 // http://www.gnu.org/licenses/lgpl-2.1.txt
20 //////////////////////////////////////////////////////////////////////////////////
24 * @brief This file contains a template class for lazy initialization
30 #include <MUtils/Global.h>
31 #include <MUtils/Exception.h>
35 #include <QAtomicPointer>
44 * \brief Lazy initialization template class
46 * The lazy-initialized value of type T can be obtained from a `Lazy<T>` instance by using the `operator*()`. Initialization of the value happens when the `operator*()` is called for the very first time, by invoking the `initializer` lambda-function that was passed to the constructor. The return value of the `initializer` lambda-function is then stored internally, so that any subsequent call to the `operator*()` *immediately* returns the previously created value.
48 * **Note on thread-saftey:** This class is thread-safe in the sense that all calls to `operator*()` on the same `Lazy<T>` instance, regardless from which thread, are guaranteed to return the exactly same value/object. The *first* thread trying to access the value will invoke the `initializer` lambda-function; concurrent threads may need to busy-wait until the initialization is completed. The `initializer` lambda-function is invoked at most once.
50 template<typename T
> class Lazy
53 Lazy(std::function
<T
*(void)> &&initializer
) : m_initializer(initializer
) { }
72 if(T
*const value
= m_value
)
79 __forceinline T
* getValue()
82 while (!(value
= m_value
))
84 if (m_state
.testAndSetOrdered(0, 1))
86 if (value
= m_initializer())
88 m_value
.fetchAndStoreOrdered(value
);
89 m_state
.fetchAndStoreOrdered(2);
92 m_state
.fetchAndStoreOrdered(0);
93 MUTILS_THROW("Initializer returned NULL pointer!");
95 QThread::yieldCurrentThread();
101 QAtomicPointer
<T
> m_value
;
103 const std::function
<T
*(void)> m_initializer
;