2024-04-25 06:45:52 +00:00
|
|
|
import threading
|
|
|
|
|
|
|
|
|
2022-03-10 03:20:04 +00:00
|
|
|
class SingletonMeta(type):
|
|
|
|
"""
|
2024-04-25 06:45:52 +00:00
|
|
|
Thread-safe Singleton Meta with double-checked locking.
|
|
|
|
Reference: https://en.wikipedia.org/wiki/Double-checked_locking
|
2022-03-10 03:20:04 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
_instances = {}
|
2024-04-25 06:45:52 +00:00
|
|
|
_lock = threading.Lock()
|
2022-03-10 03:20:04 +00:00
|
|
|
|
|
|
|
def __call__(cls, *args, **kwargs):
|
2024-04-25 06:45:52 +00:00
|
|
|
# First check (without locking) for performance reasons
|
2022-03-10 03:20:04 +00:00
|
|
|
if cls not in cls._instances:
|
2024-04-25 06:45:52 +00:00
|
|
|
# Acquire a lock before proceeding to the second check
|
|
|
|
with cls._lock:
|
|
|
|
# Second check with lock held to ensure thread safety
|
|
|
|
if cls not in cls._instances:
|
|
|
|
instance = super().__call__(*args, **kwargs)
|
|
|
|
cls._instances[cls] = instance
|
2022-09-13 10:30:18 +00:00
|
|
|
else:
|
2023-09-19 06:20:26 +00:00
|
|
|
assert (
|
|
|
|
len(args) == 0 and len(kwargs) == 0
|
2024-04-25 06:45:52 +00:00
|
|
|
), f"{cls.__name__} is a singleton class and an instance has been created."
|
|
|
|
|
2022-03-10 03:20:04 +00:00
|
|
|
return cls._instances[cls]
|