ホームページ バックエンド開発 Python チュートリアル Python でシングルトン パターンを作成する 5 つの一般的な方法

Python でシングルトン パターンを作成する 5 つの一般的な方法

Oct 18, 2016 am 09:31 AM

いわゆるシングルトンとは、クラスのインスタンスが最初から最後まで 1 回しか作成できないことを意味します。

方法 1

クラスに最初から最後まで最大 1 つのインスタンスを持たせたい場合は、__new__ メソッドを使用するのが非常に簡単です。 Python のクラスは、__new__ を通じてインスタンスを作成します:

class Singleton(object):
    def __new__(cls,*args,**kwargs):
        if not hasattr(cls,'_inst'):
            cls._inst=super(Singleton,cls).__new__(cls,*args,**kwargs)
        return cls._inst
if __name__=='__main__':
    class A(Singleton):
        def __init__(self,s):
            self.s=s      
    a=A('apple')   
    b=A('banana')
    print id(a),a.s
    print id(b),b.s
ログイン後にコピー


結果:

29922256 Banana

29922256 Banana

__new__ メソッドを使用して、クラスのインスタンスが作成されたときにそのクラスのインスタンスをバインドします。 cls._inst が None の場合、クラスがインスタンス化されていないことを意味し、インスタンスを cls._inst にバインドします。最初のインスタンス化で作成されたインスタンスは、今後インスタンス化されるたびに返されます。シングルトンからサブクラスを派生するときは、__new__ をオーバーロードしないように注意してください。

方法 2:

場合によっては、生成されたインスタンスが同じ ID を持つかどうかは気にせず、そのステータスと動作だけを気にすることがあります。多数のインスタンスの作成を許可できますが、すべての

class Borg(object):
    _shared_state={}
    def __new__(cls,*args,**kwargs):
        obj=super(Borg,cls).__new__(cls,*args,**kwargs)
        obj.__dict__=cls._shared_state
        return obj
ログイン後にコピー

は、すべてのインスタンスの __dict__ が同じ辞書を指すようにするため、インスタンスは同じメソッドとプロパティを共有します。インスタンスの name 属性の設定は、__init__ 内で変更されたか直接変更されたかに関係なく、すべてのインスタンスに影響します。ただし、インスタンス ID は異なります。クラス インスタンスが属性を共有できるが、サブクラスとは共有できないようにするには、Borg._shared_state の代わりに cls._shared_state を必ず使用してください。

インスタンスの ID が異なるため、各インスタンスを辞書キーとして使用できます:

if __name__=='__main__':
    class Example(Borg):
        pass
    a=Example()
    b=Example()
    c=Example()
    adict={}
    j=0
    for i in a,b,c:
        adict[i]=j
        j+=1
    for i in a,b,c:
        print adict[i]
ログイン後にコピー

結果:

0

1

2

この動作が希望どおりでない場合は、次のようにすることができます。 __eq__ メソッドと __hash__ メソッドを Borg クラスに追加して、シングルトン パターンの動作に近づけます。

class Borg(object):
    _shared_state={}
    def __new__(cls,*args,**kwargs):
        obj=super(Borg,cls).__new__(cls,*args,**kwargs)
        obj.__dict__=cls._shared_state
        return obj
    def __hash__(self):
        return 1
    def __eq__(self,other):
        try:
            return self.__dict__ is other.__dict__
        except:
            return False
if __name__=='__main__':
    class Example(Borg):
        pass
    a=Example()
    b=Example()
    c=Example()
    adict={}
    j=0
    for i in a,b,c:
        adict[i]=j
        j+=1
    for i in a,b,c:
        print adict[i]
ログイン後にコピー

結果:

2

2

2

キーが使用されている場合、すべてのインスタンスが可能です。

方法 3

クラスを作成すると、いくつかのメカニズムがクラス名、基本クラスのタプル、クラス ディクショナリを使用してクラス オブジェクトを作成します。新しいクラスのこのメカニズムはデフォルトで type になり、このメカニズムはプログラム可能であり、メタクラス __metaclass__ と呼ばれます。

class Singleton(type):
    def __init__(self,name,bases,class_dict):
        super(Singleton,self).__init__(name,bases,class_dict)
        self._instance=None
    def __call__(self,*args,**kwargs):
        if self._instance is None:
            self._instance=super(Singleton,self).__call__(*args,**kwargs)
        return self._instance
if __name__=='__main__':
    class A(object):
        __metaclass__=Singleton       
    a=A()
    b=A()
    print id(a),id(b)
ログイン後にコピー

結果:

34248016 34248016

IDは同じです。

この例では、Singleton メタクラスを構築し、 __call__ メソッドを使用して関数の動作をシミュレートします。クラス A を構築するときにそのメタクラスを Singleton に設定し、クラス オブジェクト A を作成するときに次のように動作します:

A=Singleton(name, baseds, class_dict), A は実際には Singleton クラスのインスタンスです。

A のインスタンスを作成するとき、A()=Singleton(name,bases,class_dict)()=Singleton(name,bases,class_dict).__call__() となり、A のすべてのインスタンスが A_ の属性を指すようになります。たとえば、この方法は実際には方法 1 と同じです。

方法 4

Python の module モジュールはプログラム内で 1 回だけロードされ、それ自体がシングルトンです。モジュールを直接記述し、モジュール内で必要なメソッドと属性をモジュール スコープ内の関数およびグローバル変数として記述することができます。クラスを記述する必要はまったくありません。

そして、モジュールとクラスの利点を組み合わせる方法がいくつかあります:

class _singleton(object):
    class ConstError(TypeError):
        pass
    def __setattr__(self,name,value):
        if name in self.__dict__:
            raise self.ConstError
        self.__dict__[name]=value
    def __delattr__(self,name):
        if name in self.__dict__:
            raise self.ConstError
        raise NameError
import sys
sys.modules[__name__]=_singleton()
ログイン後にコピー

Python は sys.modules をチェックしてモジュール オブジェクトであることを確認しません。これを使用してモジュールをクラス オブジェクトにバインドします。将来的には同じオブジェクトにバインドされます。

コードをsingle.pyに保存します:

>>> import single
>>> single.a=1
>>> single.a=2
ログイン後にコピー

ConstError

>>> del single.a

ConstError

方法5

最も簡単な方法:


りー

名前 singleton をインスタンスにバインドすると、singleton が独自のクラスの唯一のオブジェクトになります。


このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。

ホットAIツール

Undresser.AI Undress

Undresser.AI Undress

リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover

AI Clothes Remover

写真から衣服を削除するオンライン AI ツール。

Undress AI Tool

Undress AI Tool

脱衣画像を無料で

Clothoff.io

Clothoff.io

AI衣類リムーバー

AI Hentai Generator

AI Hentai Generator

AIヘンタイを無料で生成します。

ホットツール

メモ帳++7.3.1

メモ帳++7.3.1

使いやすく無料のコードエディター

SublimeText3 中国語版

SublimeText3 中国語版

中国語版、とても使いやすい

ゼンドスタジオ 13.0.1

ゼンドスタジオ 13.0.1

強力な PHP 統合開発環境

ドリームウィーバー CS6

ドリームウィーバー CS6

ビジュアル Web 開発ツール

SublimeText3 Mac版

SublimeText3 Mac版

神レベルのコード編集ソフト(SublimeText3)

LinuxターミナルでPythonバージョンを表示するときに発生する権限の問題を解決する方法は? LinuxターミナルでPythonバージョンを表示するときに発生する権限の問題を解決する方法は? Apr 01, 2025 pm 05:09 PM

LinuxターミナルでPythonバージョンを表示する際の許可の問題の解決策PythonターミナルでPythonバージョンを表示しようとするとき、Pythonを入力してください...

HTMLを解析するために美しいスープを使用するにはどうすればよいですか? HTMLを解析するために美しいスープを使用するにはどうすればよいですか? Mar 10, 2025 pm 06:54 PM

この記事では、Pythonライブラリである美しいスープを使用してHTMLを解析する方法について説明します。 find()、find_all()、select()、およびget_text()などの一般的な方法は、データ抽出、多様なHTML構造とエラーの処理、および代替案(SEL

TensorflowまたはPytorchで深い学習を実行する方法は? TensorflowまたはPytorchで深い学習を実行する方法は? Mar 10, 2025 pm 06:52 PM

この記事では、深い学習のためにTensorflowとPytorchを比較しています。 関連する手順、データの準備、モデルの構築、トレーニング、評価、展開について詳しく説明しています。 特に計算グラップに関して、フレームワーク間の重要な違い

あるデータフレームの列全体を、Python内の異なる構造を持つ別のデータフレームに効率的にコピーする方法は? あるデータフレームの列全体を、Python内の異なる構造を持つ別のデータフレームに効率的にコピーする方法は? Apr 01, 2025 pm 11:15 PM

PythonのPandasライブラリを使用する場合、異なる構造を持つ2つのデータフレーム間で列全体をコピーする方法は一般的な問題です。 2つのデータがあるとします...

Pythonでコマンドラインインターフェイス(CLI)を作成する方法は? Pythonでコマンドラインインターフェイス(CLI)を作成する方法は? Mar 10, 2025 pm 06:48 PM

この記事では、コマンドラインインターフェイス(CLI)の構築に関するPython開発者をガイドします。 Typer、Click、Argparseなどのライブラリを使用して、入力/出力の処理を強調し、CLIの使いやすさを改善するためのユーザーフレンドリーな設計パターンを促進することを詳述しています。

人気のあるPythonライブラリとその用途は何ですか? 人気のあるPythonライブラリとその用途は何ですか? Mar 21, 2025 pm 06:46 PM

この記事では、numpy、pandas、matplotlib、scikit-learn、tensorflow、django、flask、and requestsなどの人気のあるPythonライブラリについて説明し、科学的コンピューティング、データ分析、視覚化、機械学習、Web開発、Hの使用について説明します。

Pythonの仮想環境の目的を説明してください。 Pythonの仮想環境の目的を説明してください。 Mar 19, 2025 pm 02:27 PM

この記事では、Pythonにおける仮想環境の役割について説明し、プロジェクトの依存関係の管理と競合の回避に焦点を当てています。プロジェクト管理の改善と依存関係の問題を減らすための作成、アクティベーション、およびメリットを詳しく説明しています。

正規表現とは何ですか? 正規表現とは何ですか? Mar 20, 2025 pm 06:25 PM

正規表現は、プログラミングにおけるパターンマッチングとテキスト操作のための強力なツールであり、さまざまなアプリケーションにわたるテキスト処理の効率を高めます。

See all articles