python中的特殊方法 __getitem__()

特殊方法----getitem()

Python的特殊方法__getitem_() 主要作用是可以让对象实现迭代功能。我们通过一个实例来说明。

定义一个Sentence类,通过索引提取单词。

import re
RE_WORD = re.compile(r'\w+')
class Sentence:
    def __init__(self, text):
        self.text = text
        self.words = RE_WORD.findall(text)  # re.findall函数返回一个字符串列表,里面的元素是正则表达式的全部非重叠匹配
    def __getitem__(self, index):
        return self.words[index]  
  • 测试
>>> s = Sentence('The time has come')
>>> for word in s:
            print(word)
    
    The
    time
    has
    come
>>> s[0]
    'The'
>>> s[1]
    'time'

通过测试发现,示例 s 可以正常迭代。但是没有定义 getitem() 测试则会报错, TypeError: '***' object is not iterable

序列可以迭代

我们都知道序列是可以迭代,下面具体说明原因。

解释器需要迭代对象x时, 会自动调用iter(x)方法。内置的 iter(x) 方法有以下作用

  1. 检查对象是否实现了__inter__ 方法,如果实现了就调用它(也就是我们偶尔用到的特殊方法重载),获取一个迭代器。
  2. 如果没有实现iter()方法, 但是实现了 __getitem__方法,Python会创建一个迭代器,尝试按顺序(从索引0开始,可以看到我们刚才是通过s[0]取值)获取元素。
  3. 如果尝试失败,Python抛出TypeError异常,通常会提示TypeError: '***' object is not iterable

任何Python序列都可迭代的原因是,他们都实现了__getitem__方法。其实,标准的序列也都实现了 __iter__方法。

注意: 从python3.4 开始,检查对象x能否迭代,最准确的方法是: 调用iter(x)方法,如果不可迭代,在处理TypeError异常。这比使用isinstance(x,abc.Iterable)更准确,因为iter()方法会考虑到遗留的__getitem__()方法,而abc.Iterable类则不考虑。

Python,`__getitem__`是一个特殊方法,用于定义对象的索引和切片行为。当您尝试通过索引或切片访问对象时,Python将调用此方法。这通常在您正在处理如列表(list)、字典(dict)、字符串(str)等可迭代对象时使用。 当类继承自内置的序列类型(例如list、str、tuple等)并使用`__getitem__`方法时,可以在类的实例实现更复杂的数据访问操作。 这是一个基本的示例,展示如何使用`__getitem__`方法来处理一个类继承自list的自定义列表: ```python class MyList(list): def __getitem__(self, index): # 可以在这里实现一些特定的索引逻辑 print("Getting item at index", index) return super().__getitem__(index) my_list = MyList([1, 2, 3, 4]) print(my_list) # 输出: "Getting item at index 1" # 输出: 2 ``` 在这个例子,当我们通过索引访问`my_list`的元素时,Python会首先调用`__getitem__`方法,然后才调用内置的`list.__getitem__`方法。这意味着我们可以在这个方法实现一些特定的逻辑,比如打印出每次访问的索引,或者返回特殊的结果。 需要注意的是,`__getitem__`方法通常返回被索引的对象,而不是该对象本身。这是因为Python的索引操作通常期望返回的是原始对象,而不是对同一对象的引用。例如,当我们使用索引访问列表时,我们期望得到的是列表的元素,而不是对列表的引用。 此外,如果你想要实现切片操作,你可以在`__getitem__`方法处理这些情况。例如,你可以定义如何根据起始和结束索引来生成一个新的序列。例如: ```python class MyList(list): def __getitem__(self, index): if isinstance(index, slice): start, stop = index.start, index.stop return MyList(super().__getitem__(start, stop)) else: return super().__getitem__(index) ``` 在这个例子,如果尝试使用切片操作访问列表,`__getitem__`方法将返回一个新的`MyList`实例,其包含了原始列表的子集。这对于处理类似列表的数据结构非常有用。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值