Python.use(better) #Vector: step08x -- if hasattr(v2, "elements"):

記事一覧 Python.use(better)《Python3.1》

if hasattr(v2, "elements"):

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

課題を作成する過程を通して「数値演算」の理解を深めます。
※ Python1.5 で作成した例題を、Python3.1 で再構成しました。

事例:コードの解説

    class Vector(object):
        ...
        def __mul__(v1, v2):
            if hasattr(v2, "elements"):
                return sum(e1*e2
                    for e1,e2 in zip(v1.elements, v2.elements))
            else:
                s = [e*v2 for e in v1.elements]
                return Vector(*s)

    ## ----------------------------------------
    local = locals()
    ex_vector7(local)
■ #1: 組み込み関数:hasattr
            if hasattr(v2, "elements"):
                return sum(e1*e2
                    for e1,e2 in zip(v1.elements, v2.elements))

組み込み関数 hasattr は、 ベクトル v2 が属性 elements を持つかどうかを判定します。

  • ベクトル v2 が属性 elements を持つなら、内積を求めて、その値をリターン値にします。
■ #2: 内包を使って
            else:
                s = [e*v2 for e in v1.elements]
                return Vector(*s)

スカラー倍のベクトルを生成します。

  • ベクトル v1 の各要素 e を順に参照します。
  • 各要素をスカラー倍した値 e*v2 を列挙した、リスト s が得られます。
  • リスト s が保持する要素をもとに、新たなインスタンスを生成 Vector() します。

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

■ 全項目を確認する

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

$ python -i vector.py
>>> do()
 0: step00 -- class Vector(object):
 1: step01 -- def __init__(self, *args):
 2: step02x -- def __add__(v1, v2):
 3: step03 -- return Vector(*s)
 4: step04 -- def __sub__(v1, v2):
 5: step05 -- def __neg__(self):
 6: step06 -- def __mul__(v1, v2):
 7: step07x -- sum(e1*e2 ...)
 8: step08x -- if hasattr(v2, "elements"):
 9: step09 -- def __rmul__(v1, v2):
10: step10 -- def __iter__(self):
11: step11x -- def __radd__(v1, v2):
12: step12x -- raise(TypeError(s))
13: step13 -- raise(TypeError(s))
14: step14 -- def _typeError(self, v1, v2, op):
>>>
■ 各項目を実行する

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

>>> do(8)
>>> # -------------------------------------------------- step08x
>>> v  = Vector(); v
()
>>> v1 = Vector(3,4); v1
(3, 4)
>>> v2 = Vector(5,-2); v2
(5, -2)
>>> v1+v2
(8, 2)
>>> v1-v2
(-2, 6)
>>> v2-v1
(2, -6)
>>> -v1
(-3, -4)
>>> -v2
(-5, 2)
>>> v1*v2
7
>>> v2*v1
7
>>> v1*3
(9, 12)
>>> v2*(-2)
(-10, 4)
>>> 3*v1
TypeError: unsupported operand type(s) for *: 'int' and 'Vector'
>>> (-2)*v2
TypeError: unsupported operand type(s) for *: 'int' and 'Vector'
>>>

2つのインスタンス v1,v2 を生成するとともに、ベクトルの内積が得られます。

  • ベクトルに任意の整数を掛け合わると、スカラー倍したベクトルが得られます。
  • 任意の整数にベクトルを掛け合わせようとすると、例外 TypeError を生成します。
■ 何が問題か

組み込み型 int は、新たに定義した Vector の存在を知りません。そのため、利用者定義の任意の型に対して、既存の演算子を適用したいときには、注意が必要です。

《余録》テストケース

def ex_vector7(local):                        #@:
    ex_vector6(local)

    source = '''
3*v1
(-2)*v2
'''.split("\n")

    do_it(source, local)

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


関連記事

Last updated♪2009/11/21