python --- 特殊方法__len__,__getitem__

文章讲述了Python中特殊方法如`__len__`和`__getitem__`的作用机制,它们如何在内置类型和自定义类中被自动调用,以及在`FrenchDeck`类实例中的应用,包括列表推导式、切片和随机选择等操作中涉及的特殊方法调用.
摘要由CSDN通过智能技术生成
#内建的一个模块,用来构建只有少数属性,但是没有方法的对象

'''
Python 解释器碰到特殊的句法时,会使用特殊方法去激活一些基本的对
象操作,这些特殊方法的名字以两个下划线开头,以两个下划线结尾
__len__,__getitem__
首先明确一点,特殊方法的存在是为了被 Python 解释器调用的,你自己并不需要调用它
们。也就是说没有 my_object.__len__() 这种写法,而应该使用 len(my_object)。在执行
len(my_object) 的时候,如果 my_object 是一个自定义类的对象,那么 Python 会自己去调
用其中由你实现的 __len__ 方法。
然而如果是 Python 内置的类型,比如列表(list)、字符串(str)、字节序列(bytearray)
等,那么 CPython 会抄个近路,__len__ 实际上会直接返回 PyVarObject 里的 ob_size 属
性。PyVarObject 是表示内存中长度可变的内置对象的 C 语言结构体。直接读取这个值比
调用一个方法要快很多。
很多时候,特殊方法的调用是隐式的,比如 for i in x: 这个语句,背后其实用的是
iter(x),而这个函数的背后则是 x.__iter__() 方法。当然前提是这个方法在 x 中被实现了。

'''


import collections

Card = collections.namedtuple('Card', ['rank', 'suit'])

class FrenchDeck:
    ranks = [str(n) for n in range(2, 11)] + list('JQKA')
    suits = 'spades diamonds clubs hearts'.split()

    def __init__(self):
        self._cards = [Card(rank, suit) for suit in self.suits
                                        for rank in self.ranks]

    def __len__(self):
        print("call __len__")
        return len(self._cards)

    def __getitem__(self, position):
        print("call __getitem__")
        return self._cards[position]


beer_card = Card('7','diamonds')
print(beer_card) #Card(rank='7', suit='diamonds')

deck = FrenchDeck()
print(len(deck))# len() 会调用对象的__len__()方法

#[]会调用__getitem__()
print(deck[0])
print(deck[-1])

from random import choice
print('-'*30)
#choice 调用了__len__ 与 __getitem__方法
choice(deck)
#调用了__getitem__
print('切片:',deck[:3])
print('*'*50)
ret_val = Card('Q', 'hearts') in deck
print('ret_val:',ret_val)

ret_val = Card('7', 'beasts') in deck
print('ret_val:',ret_val)

suit_values = dict(spades=3, hearts=2, diamonds=1, clubs=0)


def spades_high(card):
    rank_value = FrenchDeck.ranks.index(card.rank)
    return rank_value * len(suit_values) + suit_values[card.suit]


for card in sorted(deck, key=spades_high):
    print(card)
Python 中,__getitem__ 方法是一种特殊方法,用于让对象像序列一样支持索引操作。当我们使用索引操作符 [] 来获取一个对象的元素时,Python 会自动调用该对象的 __getitem__ 方法,并将索引作为参数传递给该方法。__getitem__ 方法需要返回与索引对应的元素。 例如,我们可以定义一个自定义列表类,实现 __getitem__ 方法,使其支持索引操作: ``` class MyList: def __init__(self, data): self.data = data def __getitem__(self, index): return self.data[index] ``` 然后我们可以创建一个 MyList 对象并像使用列表一样进行索引操作: ``` my_list = MyList([1, 2, 3]) print(my_list[0]) # 输出 1 print(my_list[1]) # 输出 2 print(my_list[2]) # 输出 3 ``` 除了支持索引操作外,__getitem__ 方法还可以支持切片操作。当我们使用切片操作符 [start:end:step] 时,Python 会自动调用该对象的 __getitem__ 方法,并将切片对象作为参数传递给该方法。__getitem__ 方法需要返回与切片对象对应的元素序列。 例如,我们可以在上面的 MyList 类中添加对切片操作的支持: ``` class MyList: def __init__(self, data): self.data = data def __getitem__(self, index_or_slice): if isinstance(index_or_slice, int): return self.data[index_or_slice] else: start, stop, step = index_or_slice.indices(len(self.data)) return [self.data[i] for i in range(start, stop, step)] ``` 然后我们可以像使用列表一样进行切片操作: ``` my_list = MyList([1, 2, 3, 4, 5]) print(my_list[1:4]) # 输出 [2, 3, 4] print(my_list[::2]) # 输出 [1, 3, 5] ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一个专研技术的小蜜蜂

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值