python中getitem_解析Python中的__getitem__专有方法

__getitem__来看个简单的例子就明白:

def __getitem__(self, key): return self.data[key]

>>> f = fileinfo.FileInfo("/music/_singles/kairo.mp3")

>>> f

{'name':'/music/_singles/kairo.mp3'}

>>> f.__getitem__("name")

'/music/_singles/kairo.mp3'

>>> f["name"]

'/music/_singles/kairo.mp3'

(1)  __getitem__ 专用方法很简单。像普通的方法 clear,keys 和 values 一样,它只是重定向到字典,返回字典的值。但是怎么调用它呢?哦,你可以直接调用 __getitem__,但是在实际中你其实不会那样做:我在这里执行它只是要告诉你它是如何工作的。正确地使用 __getitem__ 的方法是让 Python 来替你调用。

(2)  这个看上去就像你用来得到一个字典值的语法,事实上它返回你期望的值。下面是隐藏起来的一个环节:暗地里Python 已经将这个语法转化为 f.__getitem__("name") 的方法调用。这就是为什么 __getitem__ 是一个专用类方法的原因,不仅仅是你可以自已调用它,还可以通过使用正确的语法让 Python 来替你调用。

使用slice切片对象list有个神奇的切片方法:

>>> range(100)[5:10]

[5, 6, 7, 8, 9]

对于Fib却报错。原因是__getitem__()传入的参数可能是一个int,也可能是一个切片对象slice,所以要做判断:

class Fib(object):

def __getitem__(self, n):

if isinstance(n, int):

a, b = 1, 1

for x in range(n):

a, b = b, a + b

return a

if isinstance(n, slice):

start = n.start

stop = n.stop

a, b = 1, 1

L = []

for x in range(stop):

if x >= start:

L.append(a)

a, b = b, a + b

return L

现在试试Fib的切片:

>>> f = Fib()

>>> f[0:5]

[1, 1, 2, 3, 5]

>>> f[:10]

[1, 1, 2, 3, 5, 8, 13, 21, 34, 55]

但是没有对step参数作处理:

>>> f[:10:2]

[1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89]

也没有对负数作处理,所以,要正确实现一个__getitem__()还是有很多工作要做的。

此外,如果把对象看成dict,__getitem__()的参数也可能是一个可以作key的object,例如str。

与之对应的是__setitem__()方法,把对象视作list或dict来对集合赋值。最后,还有一个__delitem__()方法,用于删除某个元素。

总之,通过上面的方法,我们自己定义的类表现得和Python自带的list、tuple、dict没什么区别,这完全归功于动态语言的“鸭子类型”,不需要强制继承某个接口。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
解释下面代码的作用“class Dataset(object): def __init__(self, mixture_reader, targets_reader_list): self.mixture_reader = mixture_reader self.keys_list = mixture_reader.wave_keys self.targets_reader_list = targets_reader_list def __len__(self): return len(self.keys_list) def _has_target(self, key): for targets_reader in self.targets_reader_list: if key not in targets_reader: return False return True def _index_by_key(self, key): """ Return a tuple like (matrix, [matrix, ...]) """ if key not in self.mixture_reader or not self._has_target(key): raise KeyError("Missing targets or mixture") target_list = [reader[key] for reader in self.targets_reader_list] return (self.mixture_reader[key], target_list) def _index_by_num(self, num): """ Return a tuple like (matrix, [matrix, ...]) """ if num >= len(self.keys_list): raise IndexError("Index out of dataset, {} vs {}".format( num, len(self.keys_list))) key = self.keys_list[num] return self._index_by_key(key) def _index_by_list(self, list_idx): """ Returns a list of tuple like [ (matrix, [matrix, ...]), (matrix, [matrix, ...]), ... ] """ if max(list_idx) >= len(self.keys_list): raise IndexError("Index list contains index out of dataset") return [self._index_by_num(index) for index in list_idx] def __getitem__(self, index): """ Implement to support multi-type index: by key, number or list """ if type(index) == int: return self._index_by_num(index) elif type(index) == str: return self._index_by_key(index) elif type(index) == list: return self._index_by_list(index) else: raise KeyError("Unsupported index type(int/str/list)")”
03-27
这段代码定义了一个名为“Dataset”的类,它用于处理数据集。具体而言,这个类包含了以下几个方法: 1. `__init__(self, mixture_reader, targets_reader_list)`:这个方法是类的构造函数,它接收两个参数,分别是混合音频的阅读器和目标音频的阅读器列表。它将这些参数存储在类的实例变量。 2. `__len__(self)`:这个方法返回数据集的样本数量,它使用了`self.keys_list`变量存储的键的数量来计算。 3. `_has_target(self, key)`:这个方法检查给定的键是否存在于目标音频的阅读器列表。如果存在,则返回True,否则返回False。 4. `_index_by_key(self, key)`:这个方法通过给定的键查找数据集的样本。如果找到了混合音频和目标音频,则返回一个元组,其第一个元素是混合音频的值,第二个元素是目标音频的值列表。 5. `_index_by_num(self, num)`:这个方法通过给定的索引号查找数据集的样本。如果索引号超出了数据集的范围,则会引发一个IndexError异常。如果找到了混合音频和目标音频,则返回一个元组,其第一个元素是混合音频的值,第二个元素是目标音频的值列表。 6. `_index_by_list(self, list_idx)`:这个方法通过给定的索引号列表查找数据集的样本。如果索引号超出了数据集的范围,则会引发一个IndexError异常。如果找到了混合音频和目标音频,则返回一个元组列表,其每个元组都包含混合音频的值和目标音频的值列表。 7. `__getitem__(self, index)`:这个方法是类的索引函数,它根据传入的索引类型(键、数字或列表)调用不同的方法来查找数据集的样本。如果索引类型不被支持,则会引发一个KeyError异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值