Reputation: 21991
The class that acts as my starting point is as follows:
class Test:
def __getitem__(self, key):
global frame
frame = inspect.currentframe()
if key > 9:
raise KeyError
return key
My thought was to use frame.f_back
to discover that an iterator was automatically created for an instance as in the following example:
for x in Test():
x
After running both of these and looking at frame.f_back
, it was not obvious if __getitem__
can get enough information to detect if it is being called from whatever "iterator" that is interacting with it. The easiest solution would be to make the container start at one instead of zero to access its contents or maybe even to force wrapping the key in a list before passing it to the function as shown here:
>>> class Test:
def __getitem__(self, key):
if not isinstance(key, list):
raise TypeError
if len(key) != 1:
raise ValueError
key = key.pop()
if not isinstance(key, int):
raise TypeError
if not 0 <= key < 10:
raise KeyError
return key
>>> for x in Test():
x
Traceback (most recent call last):
File "<pyshell#42>", line 1, in <module>
for x in Test():
File "<pyshell#39>", line 4, in __getitem__
raise TypeError
TypeError
>>> Test()[[5]]
5
>>>
Is there a way that __getitem__
can know that it is being used automatically as an iterator and raise an exception to prevent such usage?
Related: Why does defining __getitem__ on a class make it iterable in python?
Upvotes: 2
Views: 551
Reputation: 46
I appreciate that this is a fairly old thread, but I don't believe these answers to be quite sufficient. In particular if we define __iter__
as
class MyClass:
...
def __iter__(self):
raise TypeError
...
then Python now thinks that isinstance(myobj, collections.abc.Iterable)
is True
(since it looks for an __iter__
method, source), when it explicitly isn't!
The right solution (found here) is actually to do the following:
class MyClass:
...
__iter__ = None
...
Upvotes: 2
Reputation: 2002
The way to do this is implementing both __getitem__
and __iter__
methods.
When trying to acces individual elements __getitem__
will be triggered, when trying to iterate __iter__
will be called and an exception raised.
class Test():
def __getitem__(self, k):
return k
def __iter__(self):
raise TypeError
Upvotes: 2
Reputation: 13848
If the goal is to stop the object from being an iterable, you can just force an error on __iter__
method:
class Test:
def __getitem__(self, key):
if key > 9:
raise KeyError
return key
def __iter__(self):
raise TypeError('Object is not iterable')
Test run:
>>> t = Test()
>>> for x in t:
print(x)
Traceback (most recent call last):
File "<pyshell#126>", line 1, in <module>
for x in t:
File "<pyshell#122>", line 7, in __iter__
raise TypeError('Object is not iterable')
TypeError: Object is not iterable
But __getitem__
will still work:
>>> t[0]
0
>>> t[1]
1
>>> t[10]
Traceback (most recent call last):
File "<pyshell#129>", line 1, in <module>
t[10]
File "<pyshell#122>", line 4, in __getitem__
raise KeyError
KeyError
Upvotes: 2