Why Are __getitem__(key) And Get(key) Significantly Slower Than [key]?
Solution 1:
First, the disassembly posted by Not_a_Golfer:
>>>d = {1:2}>>>dis.dis(lambda: d[1])
1 0 LOAD_GLOBAL 0 (d)
3 LOAD_CONST 1 (1)
6 BINARY_SUBSCR
7 RETURN_VALUE
>>>dis.dis(lambda: d.get(1))
1 0 LOAD_GLOBAL 0 (d)
3 LOAD_ATTR 1 (get)
6 LOAD_CONST 1 (1)
9 CALL_FUNCTION 1
12 RETURN_VALUE
>>>dis.dis(lambda: d.__getitem__(1))
1 0 LOAD_GLOBAL 0 (d)
3 LOAD_ATTR 1 (__getitem__)
6 LOAD_CONST 1 (1)
9 CALL_FUNCTION 1
12 RETURN_VALUE
Now, getting the benchmarking right is obviously important to read anything into the results, and I don't know enough to help much there. But assuming there really is a difference (which makes sense to me), here's my guesses about why there is:
dict.get
simply "does more"; it has to check if the key is present, and if not return its second argument (which defaults toNone
). This means there's some form of conditional or exception-catching, so I am completely unsurprised that this would have different timing characteristics to the more basic operation of retrieving the value associated with a key.Python has a specific bytecode for the "subscription" operation (as demonstrated in the disassembly). The builtin types, including
dict
, are implemented primarily in C and their implementations do not necessarily play by the normal Python rules (only their interfaces are required to, and there are plenty of corner cases even there). So my guess would be that the implementation of theBINARY_SUBSCR
opcode goes more-or-less directly to the underlying C implementations of builtin types that support this operation. For these types, I expect that it is actually__getitem__
that exists as a Python-level method to wrap the C implementation, rather than that the bracket syntax invokes the Python-level method.
It might be interesting to benchmark thing.__getitem__(key)
against thing[key]
for an instance of a custom class that implements __getitem__
; you might actually see the opposite results there as the BINARY_SUBSCR
op-code would internally have to fall back to doing equivalent work to looking up the method and invoking it.
Post a Comment for "Why Are __getitem__(key) And Get(key) Significantly Slower Than [key]?"