파이썬에서 싱글톤 구현하기

싱글톤은 어떤 클래스의 인스턴스가 하나만 만들어지도록 강제하는 디자인 패턴의 한 종류입니다. 싱글톤은 시스템 전체에 걸쳐 단 하나의 객체 인스턴스만 있으면 될 때 사용할 수 있으며, 전역 변수를 사용하지 않고도 특정한 객체를 어디서나 참조할 수 있게 되는 장점이 있습니다. 싱글톤을 구현하는 일반적인 방법으로는 1) 생성자를 private하게 만들어서 다른 객체가 직접적으로 호출할 수 없도록 하고, 2) 별도의 정적 메소드를 제공하여 싱글톤 객체를 획득할 수 있게 합니다.

싱글톤 구현방법

파이썬에서는 생성자를 숨길 수 없기 때문에, 생성자 그 자체가 싱글톤 객체를 반환하도록 하면 됩니다. 이를 구현하는 방법에는 여러가지가 있을 수 있지만, 대표적으로는 메타 클래스를 사용하는 방법이 있습니다. 생성자가 호출될 때, 싱글톤 객체를 반환하는 기본 구현을 만들어 두고, 이를 적용할 클래스에서 해당 메타 클래스를 사용하도록 하는 방식입니다. 예제는 아래와 같습니다.

# 싱글톤 구현을 위한 메타클래스
class Singleton(type)
    _instances = {}

    def __call__(cls, *args, **kwds):
        # Foo() 와 같이 호출될 때...
        if cls is not cls._instances:
            cls._instances[cls] = super().__call__(*args, **kwds)
        return cls._instances[cls]


class Foo(BaseClass, metaclass=Singleton):
    def __init__(self):
        self.x = 10

f = Foo()
g = Foo()  # 값 11은 무시되고 앞에서 만들어진 값이 표시
print(id(f), f.x)
print(id(g), g.x)

Foo()와 같이 클래스 이름의 생성자를 사용하면 해당 클래스의 내부에서는 __new__() 메소드가 호출됩니다. 따라서 메타클래스를 사용하지 않고 단순하게 작성하려면 다음과 같은 방식으로 작성하면 됩니다.

class Singleton:
    _instance = None
    def __new__(cls, *args, **kwds):
        if not isinstance(cls._instance, cls):
            cls._instance = object.__new__(cls, *args, **kwds)
        return cls._instance

    def __init__(self, *args, **kwds):
        self.x = 10

스레드 안전한 싱글톤

싱글톤을 구현하는 방법은 생각보다 간단합니다. 하지만 이 코드는 멀티스레드 환경에서 안전하게 사용할 수 있는 것은 아닙니다. 여전히 두 개 이상의 스레드에서 거의 동시에 생성자가 호출된다면 2개 이상의 인스턴스 객체가 만들어질 수도 있습니다. 따라서 스레드 안전한 싱글톤을 구현하기 위해서는 Lock 과 같은 동기화 수단이 필요합니다. 해당 클래스의 정적 속성으로 락 객체를 하나 만들어 놓고, 인스턴스를 체크, 생성하는 코드를 락의 크리티컬 구간 내에 위치시키면 됩니다.

from threading import Lock

class Singleton:
    _instance = None
    _lock = Lock()

    def __new__(cls, *args, **kwds):
        with cls._lock:
            if not isinstance(cls._instance, cls):
            cls._instance = object.__new__(cls, *args, **kwds)
        return cls._instance

    def __init__(self, *args, **kwds):
        self.x = 10

  

Read more

워드프레스에서 고스트로 이전

워드프레스에서 고스트로 이전

이 글을 쓰면서도 믿기 힘든 사실인데, 블로그라는 걸 처음 시작한지가 20년이 되었습니다. 이글루스에서 처음 시작했다가, SK컴즈가 인수한다고 발표함과 동시에 워드프레스로 플랫폼을 옮겼죠. 워드프레스오 옮긴 이후에는 호스팅 환경을 이리 저리 옮기긴 했지만 거의 18년 가까이 워드프레스를 사용해온 것 같습니다. 그 동안 워드프레스는 블로깅 툴에서 명실상부한 범용CMS로 발전했습니다. 사실 웬만한 홈페이지들은 이제

By sooop
띄어쓰기에 대한 생각

띄어쓰기에 대한 생각

업무 메일을 쓸 때 가장 많이 쓰는 말 중에 하나가 메일 말미에 ‘업무에 참고 부탁 드립니다.‘인데요, 어느 날부터 아웃룩에서 이 ‘부탁 드립니다’가 틀렸다고 맞춤법 지적을 하기 시작했습니다. 맞는 말은 ‘부탁드립니다’라고 붙여 쓰는 거라고. 사실 아래아한글 시절부터 이전의 MS워드까지, 워드프로세서들의 한국어 맞춤법 검사 실력은 거의 있으나 마나 한

By sooop

구글 포토에서 아이클라우드로 탈출한 후기

한 때 구글 포토가 백업 용량을 무제한으로 제공해 주겠다고해서, 구글 포토를 사용해서 사진을 백업해왔습니다. 물론 이 이야기의 결말은 저나 이 글을 읽고 있는 여러분이나 모두 알고 있습니다. 사실 AI에게 학습 시킬 이미지 데이터를 모으기 위한 것일 뿐이라거나 하는 이야기는 그 당시에도 있었습니다만, 에이 그래도 구글인데 용량은 넉넉하게 주겠지…하는 순진한

By sooop

Julia의 함수 사용팁

연산자의 함수적 표기 Julia의 연산자는 기본적으로 함수이며, 함수 호출 표기와 같은 방식으로 호출하는 것이 가능합니다. 또한 그 자체로 함수이기 때문에 filter(), map() 과 같이 함수를 인자로 받는 함수에도 연산자를 그대로 적용하는 것이 가능합니다. 특히 + 연산자는 sum() 함수와 같이 여러 인자를 받아 인자들의 합을 구할 수 있습니다. 2 + 3 # = 5 +(2,

By sooop