
接下来这个为啥要叫魔法方法呢,额,这个嘛我是跟小甲鱼的视频取的名字一样的,因为会讲比较多杂的东西,有。。。
魔法方法详细阅读地址:http://bbs.fishc.com/thread-48793-1-1.html
啥是魔法方法呢?
魔法方法是“面向对象”python的一切,一般被双下划线包围着,例如__init__(self[,...]);
__init__这个是一个构造函数,在类里面一般用于属性初始化,返回值为Null,不要在__init__函数下return值。
我们在实例化对象的时候,大部分的人会觉得调用的第一个魔法方法应该是__init__,但其实不是,第一个被调用的应该是__new__(cls[,...])(注意第一个参数是类class),返回一个实例对象,通常返回是这个class类的对象(也可以返回其它对象),举个例子看下:
下面还有一个析构函数,也就是__del__(self),注意:当所有对这个对象的引用都被删除时(也就是最后一个引用被删除时),就会启动垃圾回收机制,就会调用__del__方法。
注意:只有最后引用都删除了才会调用del方法!
下面给大家简单介绍下魔法方法中属性访问:
上面是4个类的魔法方法,下面举个例子大家看下
class Att():
def __getattribute__(self, name):
print("__getattribute__")
return super().__getattribute__(name) #调用基类的__getattribute__ def __getattr__(self, name):
print("__getattr__") def __setattr__(self, name, value):
print("__setattr__")
super().__setattr__(name, value) def __delattr__(self,name):
print("__delattr__")
super().__delattr__(name)
调用结果如下:
>>> a = Att()
>>> a.x #x属性之前没有,因此调用顺序如下
__getattribute__
__getattr__
>>> a.x = 1
__setattr__
>>> a.x #x属性存在后,getattr就不会被调用了
__getattribute__
1
>>> del a.x
__delattr__
描述符的介绍:
描述符是一种具有“捆绑行为”的对象属性。访问(获取、设置和删除)它的属性时,实际是调用特殊的方法(_get_(),_set_(),_delete_())。也就是说,如果一个对象定义了这三种方法的任何一种,它就是一个描述符。
定制序列->容器
容器类型的协议
如果说你希望定制的容器是不可变的话,你只需要定义__len__()和__getitem__()方法。
如果你希望定制的容器是可变的话,除了__len__()和__getitem__()方法,你还需要定义__setitem__()和__delitem__()两个方法。
题目:定制一个列表,要求记录列表中每个元素被访问的次数。要求支持append()等原生列表的方法 要求1:实现获取、设置和删除一个元素的行为(删除一个元素的时候对应的计数器也会被删除)
要求2:增加counter(index)方法,返回index参数所指定的元素记录的访问次数
要求3:实现append()、pop()、remove()、insert()、clear()和reverse()方法(重写这些方法的时候注意考虑计数器对应的变化)
代码如下:
class Countlist(list):
def __init__(self, *args):
super().__init__(args)
self.count = []
for i in args:
self.count.append(0) def __len__(self):
return len(self.count) def __getitem__(self, key):
self.count[key] += 1
return super().__getitem__(key) def __setitem__(self, key, value):
self.count[key] += 1
super().__setitem__(key, value) def __delitem__(self, key):
del self.count[key]
super().__delitem__(key) def counter(self, key):
return self.count[key] def append(self, value):
self.count.append(0)
super().append(value) def pop(self, key=-1):
del self.count[key]
return super().pop(key) def remove(self, value):
key = super().index(value)
del self.count[key]
super().remove(value) def insert(self, key, value):
self.count.insert(key, 0)
super().insert(key, value) def clear(self):
self.count.clear()
super().clear() def reverse(self):
self.count.reverse()
super().reverse(self)
迭代器(迭代的容器,有例如序列、列表、元组、字典等)
迭代操作提供了两个内置函数
- iter() 将对象变成迭代器
- next() 迭代器返回下一个值(如果没有值了,就会抛出异常StopIteration)
举个例子大家看下:
string = 'fishc'
it = iter(string)
while True:
try:
each = next(it)
print(each)
except StopIteration:
break
结果如下:
迭代器的魔法方法如下:
- __iter__() (容器如果是迭代器,就必须实现__iter__这个魔法方法,返回迭代器本身,return self)
- __next__()(决定了迭代器的规则)
举个例子:
题目:斐波那契数列 代码如下
class Fibl:
def __init__(self, n = 20):
self.a = 0
self.b = 1
self.n = n
def __iter__(self):
return self
def __next__(self):
self.a, self.b = self.b, self.a + self.b
if self.a >= self.n:
raise StopIteration
return self.a
结果如下:
关于迭代器的题目:
要求自己写一个MyRev类,功能与reversed()相同(内置函数reversed(seq)是返回一个迭代器,是序列seq的逆序显示),例如:
代码如下:
class MyRev:
def __init__(self, value):
self.value = value
self.index = len(value) def __iter__(self):
return self def __next__(self):
if self.index == 0:
raise StopIteration self.index -= 1
return self.value[self.index]