このようにして作成したスタックは、必要に応じて、メソッドや特殊メソッドをオーバーライドして、その動作を自分のクラスに適したものに変更できる。以下では幾つか例を見てみよう。
例えば、__init__メソッドのオーバーライドについて考えてみよう。
list関数は「反復可能オブジェクトを0個または1個だけ引数に取る」のが仕様だ。
mylist = list([1, 2, 3]) # 1、2、3を要素とするリストを生成
print(mylist)
mylist = list(1, 2, 3) # エラー(引数は0個か1個だけ)
現在のMyStack3クラスでは__init__メソッドを定義していないので、そのインスタンスを生成する際には基底クラスの__init__メソッドが呼び出され(て、そこにMyStack関数呼び出しに渡した引数が渡され)る。
mystack = MyStack3([1, 2, 3]) # 1、2、3を要素とするスタックを生成
print(mystack)
mystack = MyStack(1, 2, 3) # エラー(引数は0個か1個だけ)
だが、前々回に作成したスタックでは、そのインスタンス生成時に次のような振る舞いをしていた。
「スタックはリスト(の一種)」であることや、tuple関数やset関数などでも反復可能オブジェクトを受け取り、そこから要素を反復的に取り出すのが一般的であるため、現在の振る舞いの方が「よりPythonのオブジェクトらしい」といえるが、ここでは上記の振る舞いとなるように__init__メソッドをオーバーライドしてみよう。
といっても書かなければならないコードはシンプルだ。
class MyStack3(list):
def __init__(self, *args):
# print(args) # 可変長位置引数を確認したければコメントアウト
super().__init__(args)
def push(self, item):
self.append(item)
__init__メソッドには可変長引数を受け取るパラメーターを持たせる。このパラメーターには0個以上の引数が「タプル」にまとめられて渡される。以下の表は「MyStack3(……)」呼び出しの引数によってこのパラメーターがどうなるかをまとめたものだ。
MyStack3()呼び出し | パラメーターargsの値 |
---|---|
MyStack3() | ():空のタプル |
MyStack3(1) | (1,):要素が1つのタプル |
MyStack3([1, 2, 3]) | ([1, 2, 3],):要素が1つのタプル |
MyStack3(1, 2, 3) | (1, 2, 3):要素が3つのタプル |
MyStack3関数呼び出しの形式とargs |
パラメーターargsはタプル、すなわち反復可能オブジェクトなので、実はこれをそのまま基底クラスの__init__メソッドに渡せば、その要素を使ってリストが作成される。「MyStack3([1, 2, 3])」という呼び出しならタプルには要素が1つだけなので、「[1, 2, 3]」というリストがそのままスタックの要素となるし、「MyStack3(1, 2, 3)」ならタプルの要素は3つあるので、1、2、3を個々の要素とするスタックが作られるようになるわけだ。
Copyright© Digital Advantage Corp. All Rights Reserved.