Python.use(better) #クラスを用いない OOP 5/7

記事一覧 Python.use(better)《復刻版》

step04 -- def __getattr__(self, name):

《著》小粒ちゃん+∞《監修》小泉ひよ子とタマゴ倶楽部
第0版♪2001/03/02 ● 第1版♪2003/05/25 ● 第2版♪2004/06/01 ● 第3版♪2009/02/28

クラスを用いない OOP を実践することで、OOP の理解を深めます。

※ Python1.5 で作成した例題を、Python3.1 で再構成しました。

事例:モジュールを起動する

■ 全項目を確認する

全ステップの「項目」を確認するには、関数 do を利用します。

$ python -i stack_func.py
>>> do()
0: step00 -- def Stack():
1: step01 -- def push(self, item):
2: step02 -- def pop(self):
3: step03 -- def __repr__(self):
4: step04 -- def __getattr__(self, name):
>>>
■ 各項目を実行する

各ステップの「動作」を確認するには、関数 do に実引数を指定します。

>>> do(4)
>>> # -------------------------------------------------- step04
>>> s = Stack(); s

>>> s.push("A"), s
(None, ['A'])
>>> s.push("B"), s
(None, ['A', 'B'])
>>> s.pop(), s
('B', ['A'])
>>> s.pop(), s
('A', )
>>> s.pop(), s
IndexError: pop from empty list
(None, [])
>>>
  • 空のスタック s を用意します。
  • push 操作でスタック s に要素を積むとともに、None がリターン値として得られます。
  • pop 操作でスタック s から要素を降ろすとともに、その要素がリターン値として得られます。
  • 空のスタックに pop 操作を適用しようとすると、例外 IndexError を生成します。

事例:コードの解説

辞書を利用して、インスタンス属性を管理します。

    def Stack():

        class Tips(object):
            def __repr__(self):
                return self.__repr()
            def __getattr__(self, name):             #1:
                return self.__dict__["class"][name]

        self = Tips(); del Tips
        self._Tips__repr = lambda: repr(self.items)

        def class_(self):                            #2:
            return {
                "items": [],
                "push" : lambda item: push(self, item),
                "pop"  : lambda     : pop (self),
                }

        self.__dict__["class"] = class_(self)        #3:
        return self

    def push(self, item):
        self.items.append(item)

    def pop(self):
        try:
            return self.items.pop()
        except IndexError as error:
            print("IndexError:", error)
■ #1, 3: インスタンス属性の参照
            def __getattr__(self, name):             #1:
                return self.__dict__["class"][name]
        ...
        self.__dict__["class"] = class_(self)        #3:

メソッド __getattr__ は、演算子 . に呼応して、インスタンス属性を参照します。

■ #2: インスタンス属性の辞書
        def class_(self):                            #2:
            return {
                "items": [],
                "push" : lambda item: push(self, item),
                "pop"  : lambda     : pop (self),
                }

インスタンス属性を管理する辞書を作成します。

》こちらに移動中です《
TOP


関連記事

Last updated♪2009/10/27

*1:C言語で学ぶ OOP -- use(C,better), 1988, 1992, 1993.