Files
phs_v1.0.1.0/commonlibrary/c_utils/base/include/singleton.h

212 lines
5.9 KiB
C
Raw Normal View History

2024-09-27 19:16:49 +08:00
/*
* Copyright (c) 2021 Huawei Device Co., Ltd.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef UTILS_BASE_SINGLETON_H
#define UTILS_BASE_SINGLETON_H
#include "nocopyable.h"
#include <mutex>
#include <memory>
namespace OHOS {
/**
* The purpose of the following macro definitions is to reduce the need
* to write repetitive code when defining singleton classes on the client side.
*
* Taking DelayedSingleton as an example, when declaring the target class as a
* singleton, add the DECLARE_DELAYED_SINGLETON (class_name) to the class
* declaration. When using the target singleton, call
* class_name::GetInstance()->.
*/
/**
* @brief Sets `MyClass` as a `DelayedSingleton`.
*
* A `MyClass` object can be obtained by calling
* `DelayedSingleton<MyClass>::GetInstance()`.
*
* @param MyClass Target class to be set as a singleton.
*
* @note This macro definition should be used in the body of a class definition.
*/
#define DECLARE_DELAYED_SINGLETON(MyClass)\
public:\
~MyClass();\
private:\
friend DelayedSingleton<MyClass>;\
MyClass();
/**
* @brief Sets `MyClass` as a `DelayedRefSingleton`.
*
* A `MyClass` object can be obtained by calling
* `DelayedRefSingleton<MyClass>::GetInstance()`.
*
* @param MyClass Target class to be set as a singleton.
*
* @note This macro definition should be used in the body of a class definition.
*/
#define DECLARE_DELAYED_REF_SINGLETON(MyClass)\
private:\
friend DelayedRefSingleton<MyClass>;\
~MyClass();\
MyClass();
/**
* @brief Sets `MyClass` as a `Singleton`.
*
* A `MyClass` object can be obtained by calling
* `Singleton<MyClass>::GetInstance()`.
*
* @param MyClass Target class to be set as a singleton.
*
* @note This macro definition should be used in the body of a class definition.
*/
#define DECLARE_SINGLETON(MyClass)\
private:\
friend Singleton<MyClass>;\
MyClass& operator=(const MyClass&) = delete;\
MyClass(const MyClass&) = delete;\
MyClass();\
~MyClass();
/**
* @brief DelayedSingleton is a thread-safe, memory-safe lazy initialized
* singleton (with smart pointer and lock).
*/
template<typename T>
class DelayedSingleton : public NoCopyable {
public:
/**
* @brief Creates a unique instance object.
*
* Use a smart pointer to manage resources. If all shared_ptrs are
* destroyed, the new object will also be deleted. This avoids memory leaks.
* A lock is added only when the pointer is empty, so as to avoid locking
* every time `GetInstance()` is called, reducing overhead of the lock.
*/
static std::shared_ptr<T> GetInstance();
/**
* @brief Releases the ownership of managed object of the smart pointer.
*
* @note After this API is called successfully, 'GetInstance()' will create
* a new object. If the old object has an external 'std::shared_ptr'
* reference, you need to release it to maintain a singleton.
*/
static void DestroyInstance();
private:
static std::shared_ptr<T> instance_; // Record the created DelayedSingleton instance.
static std::mutex mutex_; // Mutex, which guarantees that only one thread is accessing a common resource at any time.
};
template<typename T>
std::shared_ptr<T> DelayedSingleton<T>::instance_ = nullptr;
template<typename T>
std::mutex DelayedSingleton<T>::mutex_;
template<typename T>
std::shared_ptr<T> DelayedSingleton<T>::GetInstance()
{
if (instance_ == nullptr) {
std::lock_guard<std::mutex> lock(mutex_);
if (instance_ == nullptr) {
std::shared_ptr<T> temp(new (std::nothrow) T);
instance_ = temp;
}
}
return instance_;
}
template<typename T>
void DelayedSingleton<T>::DestroyInstance()
{
std::lock_guard<std::mutex> lock(mutex_);
if (instance_ != nullptr) {
instance_.reset();
instance_ = nullptr;
}
}
/**
* @brief DelayedRefSingleton is a thread-safe, lazy initialized
* singleton (with ordinary pointer and lock).
*/
template<typename T>
class DelayedRefSingleton : public NoCopyable {
public:
/**
* @brief Creates a unique instance object.
*
* Pointer is used in the implementation, and the return value is a
* reference. It points to an instance that has a lifetime managed by
* the non-user code.
*
* @note The instance may not have been created at a certain point in time,
* or it may have been deleted, and therefore it is not possible to
* prohibit use of the delete keyword to destroy the object in advance.
*/
static T& GetInstance();
private:
static T* instance_; // Record the DelayedRefSingleton instance created.
static std::mutex mutex_;
};
template<typename T>
T* DelayedRefSingleton<T>::instance_ = nullptr;
template<typename T>
std::mutex DelayedRefSingleton<T>::mutex_;
template<typename T>
T& DelayedRefSingleton<T>::GetInstance()
{
if (instance_ == nullptr) {
std::lock_guard<std::mutex> lock(mutex_);
if (instance_ == nullptr) {
instance_ = new T();
}
}
return *instance_;
}
/**
* @brief Singleton is a normally initialized singleton (without pointers and
* locks).
*/
template<typename T>
class Singleton : public NoCopyable {
public:
/**
* @brief Gets a singleton instance object.
*/
static T& GetInstance() { return instance_; }
private:
static T instance_;
};
template<typename T>
T Singleton<T>::instance_;
} // namespace OHOS
#endif