10.3. operator
--- 関数形式の標準演算子¶
ソースコード: Lib/operator.py
operator
モジュールは、Python の組み込み演算子に対応する効率的な関数群を提供します。
例えば、 operator.add(x, y)
は式 x+y
と等価です。
多くの関数名は、特殊メソッドに使われている名前から前後の二重アンダースコアを除いたものと同じです。
後方互換性のため、ほとんどの関数に二重アンダースコアを付けたままのバージョンがあります。
簡潔さのために、二重アンダースコアが無いバージョンの方が好まれます。
これらの関数は、オブジェクト比較、論理演算、数学演算、シーケンス演算をするものに分類されます。
オブジェクト比較関数は全てのオブジェクトで有効で、関数の名前はサポートする拡張比較演算子からとられています:
-
operator.
lt
(a, b)¶ -
operator.
le
(a, b)¶ -
operator.
eq
(a, b)¶ -
operator.
ne
(a, b)¶ -
operator.
ge
(a, b)¶ -
operator.
gt
(a, b)¶ -
operator.
__lt__
(a, b)¶ -
operator.
__le__
(a, b)¶ -
operator.
__eq__
(a, b)¶ -
operator.
__ne__
(a, b)¶ -
operator.
__ge__
(a, b)¶ -
operator.
__gt__
(a, b)¶ a と b の "拡張比較 (rich comparisons)" を行います。具体的には、
lt(a, b)
はa < b
、le(a, b)
はa <= b
、eq(a, b)
はa == b
、ne(a, b)
はa != b
、gt(a, b)
はa > b
、そしてge(a, b)
はa >= b
と等価です。これらの関数はどのような値を返してもよく、ブール値として解釈できてもできなくてもかまいません。拡張比較の詳細については 比較 を参照してください。
論理演算もまた全てのオブジェクトに対して適用でき、真理値判定、同一性判定およびブール演算をサポートします:
-
operator.
not_
(obj)¶ -
operator.
__not__
(obj)¶ not
obj の結果を返します。(オブジェクトインスタンスには__not__()
メソッドは無いので注意してください; インタプリタコアがこの演算を定義しているだけです。結果は__bool__()
および__len__()
メソッドに影響されます。)
-
operator.
is_
(a, b)¶ a is b
を返します。オブジェクトの同一性を判定します。
-
operator.
is_not
(a, b)¶ a is not b
を返します。オブジェクトの同一性を判定します。
演算子で最も多いのは数学演算およびビット単位の演算です:
-
operator.
inv
(obj)¶ -
operator.
invert
(obj)¶ -
operator.
__inv__
(obj)¶ -
operator.
__invert__
(obj)¶ obj のビット単位反転を返します。
~obj
と同じです。
-
operator.
truediv
(a, b)¶ -
operator.
__truediv__
(a, b)¶ 2/3 が 0 ではなく 0.66 となるような
a / b
を返します。 "真の" 除算としても知られています。
シーケンスを扱う演算子(いくつかの演算子はマッピングも扱います)には以下のようなものがあります:
-
operator.
countOf
(a, b)¶ a の中に b が出現する回数を返します。
-
operator.
indexOf
(a, b)¶ a で最初に b が出現する場所のインデクスを返します。
-
operator.
length_hint
(obj, default=0)¶ オブジェクト o の概算の長さを返します。最初に実際の長さを、次に
object.__length_hint__()
を使って概算の長さを、そして最後にデフォルトの値を返そうとします。バージョン 3.4 で追加.
operator
モジュールは属性とアイテムの汎用的な検索のための道具も定義しています。 map()
, sorted()
, itertools.groupby()
, や関数を引数に取るその他の関数に対して高速にフィールドを抽出する際に引数として使うと便利です。
-
operator.
attrgetter
(attr)¶ -
operator.
attrgetter
(*attrs) 演算対象から attr を取得する呼び出し可能なオブジェクトを返します。二つ以上の属性を要求された場合には、属性のタプルを返します。属性名はドットを含むこともできます。例えば:
f = attrgetter('name')
とした後で、f(b)
を呼び出すとb.name
を返します。f = attrgetter('name', 'date')
とした後で、f(b)
を呼び出すと(b.name, b.date)
を返します。f = attrgetter('name.first', 'name.last')
とした後で、f(b)
を呼び出すと(b.name.first, b.name.last)
を返します。
次と等価です:
def attrgetter(*items): if any(not isinstance(item, str) for item in items): raise TypeError('attribute name must be a string') if len(items) == 1: attr = items[0] def g(obj): return resolve_attr(obj, attr) else: def g(obj): return tuple(resolve_attr(obj, attr) for attr in items) return g def resolve_attr(obj, attr): for name in attr.split("."): obj = getattr(obj, name) return obj
-
operator.
itemgetter
(item)¶ -
operator.
itemgetter
(*items) 演算対象からその
__getitem__()
メソッドを使って item を取得する呼び出し可能なオブジェクトを返します。 二つ以上のアイテムを要求された場合には、アイテムのタプルを返します。例えば:f = itemgetter(2)
とした後で、f(r)
を呼び出すとr[2]
を返します。g = itemgetter(2, 5, 3)
とした後で、g(r)
を呼び出すと(r[2], r[5], r[3])
を返します。
次と等価です:
def itemgetter(*items): if len(items) == 1: item = items[0] def g(obj): return obj[item] else: def g(obj): return tuple(obj[item] for item in items) return g
アイテムは被演算子の
__getitem__()
メソッドが受け付けるどんな型でも構いません。辞書ならば任意のハッシュ可能な値を受け付けます。リスト、タプル、文字列などはインデクスかスライスを受け付けます:>>> itemgetter(1)('ABCDEFG') 'B' >>> itemgetter(1,3,5)('ABCDEFG') ('B', 'D', 'F') >>> itemgetter(slice(2,None))('ABCDEFG') 'CDEFG'
itemgetter()
を使って特定のフィールドをタプルレコードから取り出す例:>>> inventory = [('apple', 3), ('banana', 2), ('pear', 5), ('orange', 1)] >>> getcount = itemgetter(1) >>> list(map(getcount, inventory)) [3, 2, 5, 1] >>> sorted(inventory, key=getcount) [('orange', 1), ('banana', 2), ('apple', 3), ('pear', 5)]
-
operator.
methodcaller
(name[, args...])¶ 引数の name メソッドを呼び出す呼び出し可能オブジェクトを返します。追加の引数および/またはキーワード引数が与えられると、これらもそのメソッドに引き渡されます。例えば:
f = methodcaller('name')
とした後で、f(b)
を呼び出すとb.name()
を返します。f = methodcaller('name', 'foo', bar=1)
とした後で、f(b)
を呼び出すとb.name('foo', bar=1)
を返します。
次と等価です:
def methodcaller(name, *args, **kwargs): def caller(obj): return getattr(obj, name)(*args, **kwargs) return caller
10.3.1. 演算子から関数への対応表¶
下のテーブルでは、個々の抽象的な操作が、どのように Python 構文上の各演算子や operator
モジュールの関数に対応しているかを示しています。
演算 | 操作 | 関数 |
---|---|---|
加算 | a + b |
add(a, b) |
結合 | seq1 + seq2 |
concat(seq1, seq2) |
包含判定 | obj in seq |
contains(seq, obj) |
除算 | a / b |
truediv(a, b) |
除算 | a // b |
floordiv(a, b) |
ビット単位論理積 | a & b |
and_(a, b) |
ビット単位排他的論理和 | a ^ b |
xor(a, b) |
ビット単位反転 | ~ a |
invert(a) |
ビット単位論理和 | a | b |
or_(a, b) |
冪乗 | a ** b |
pow(a, b) |
同一性 | a is b |
is_(a, b) |
同一性 | a is not b |
is_not(a, b) |
インデクス指定の代入 | obj[k] = v |
setitem(obj, k, v) |
インデクス指定の削除 | del obj[k] |
delitem(obj, k) |
インデクス指定 | obj[k] |
getitem(obj, k) |
左シフト | a << b |
lshift(a, b) |
剰余 | a % b |
mod(a, b) |
乗算 | a * b |
mul(a, b) |
行列の乗算 | a @ b |
matmul(a, b) |
(算術) 負 | - a |
neg(a) |
(論理) 否 | not a |
not_(a) |
正 | + a |
pos(a) |
右シフト | a >> b |
rshift(a, b) |
スライス指定の代入 | seq[i:j] = values |
setitem(seq, slice(i, j), values) |
スライス指定の削除 | del seq[i:j] |
delitem(seq, slice(i, j)) |
スライス指定 | seq[i:j] |
getitem(seq, slice(i, j)) |
文字列書式化 | s % obj |
mod(s, obj) |
減算 | a - b |
sub(a, b) |
真理値判定 | obj |
truth(obj) |
順序付け | a < b |
lt(a, b) |
順序付け | a <= b |
le(a, b) |
等価性 | a == b |
eq(a, b) |
不等性 | a != b |
ne(a, b) |
順序付け | a >= b |
ge(a, b) |
順序付け | a > b |
gt(a, b) |
10.3.2. インプレース演算子¶
多くの演算に「インプレース」版があります。 以下の関数はそうした演算子の通常の文法に比べてより素朴な呼び出し方を提供します。たとえば、文(statement) x += y
は x = operator.iadd(x, y)
と等価です。別の言い方をすると、 z = operator.iadd(x, y)
は複合文 z = x; z += y
と等価です。
なお、これらの例では、インプレースメソッドが呼び出されたとき、計算と代入は二段階に分けて行われます。以下に挙げるインプレース関数は、インプレースメソッドを呼び出してその第一段階だけを行います。第二段階の代入は扱われません。
文字列、数、タプルのようなイミュータブルなターゲットでは、更新された値が計算されますが、入力変数に代入し返されはしません。
>>> a = 'hello'
>>> iadd(a, ' world')
'hello world'
>>> a
'hello'
リストや辞書のようなミュータブルなターゲットでは、インプレースメソッドは更新を行うので、続く代入は必要ありません。
>>> s = ['h', 'e', 'l', 'l', 'o']
>>> iadd(s, [' ', 'w', 'o', 'r', 'l', 'd'])
['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd']
>>> s
['h', 'e', 'l', 'l', 'o', ' ', 'w', 'o', 'r', 'l', 'd']
-
operator.
iconcat
(a, b)¶ -
operator.
__iconcat__
(a, b)¶ a = iconcat(a, b)
は二つのシーケンス a と b に対しa += b
と等価です。