目录:
一、笔记
二、我的自学路线
三、笔记目录
一、笔记
1) 魔术方法
① 魔术方法就是一个类中的方法,和普通方法唯一的不同是普通方法需要调用,而魔术方法是在特定时刻自动触发。
② 这些魔术方法的名字特定,不能更改,但是入口参数的名字可以自己命名。
2) 基本的魔术方法
① __new__(cls[,...]),__new__ 是在一个对象实例化的时候所调用的第一个方法,而不是init是第一个方法,new方法返回的是对象
② __init__(self[,...]) 构造器,当一个实例被创建的时候调用的初始化方法
③ __del__(self) 构造器,当一个实例被销毁的时候调用的方法
④ __call__(self,[,args...]) 允许一个类的实例像函数一样被调用:x(a,b) 调用 x.__call__(a,b),如果没有__call__函数,不能把实例像函数一样调用,而要用x.function(a,b)调用实例x的实例化方法
⑤ __len__(self) 定义当被len()调用时的行为
class Foo(object):
# __new__是一个魔术方法,并且是实例化对象时第一个调用的方法,返回的结果是实例化对象。
def __new__(cls,*agrs,**cwds):
print('调用__new__方法')
inst = object.__new__(cls,*agrs,**cwds)
print(inst) # 返回的是一个对象
return inst
# __init__的第一个参数self是__new__魔术方法返回的结果
def __init__(self,price = 50):
print('调用__init__方法')
print(self) # self是__new__方法返回的东西,因此self命名为其他名称也可以接受__new__返回的对象
self.price = price
def how_much_of_book(self,n):
print('调用how_much_of_book方法')
print(self)
return self.price * n
foo = Foo() # 实例化Foo()返回一个对象
print(foo.how_much_of_book(8))
运行结果:
调用__new__方法
调用__init__方法
调用how_much_of_book方法
400
class Foo(object):
# __new__是一个魔术方法,并且是实例化对象时第一个调用的方法,返回的结果是实例化对象。
def __new__(cls,*agrs,**cwds):
print('调用__new__方法')
inst = object.__new__(cls,*agrs,**cwds)
print(inst) # 返回的是一个对象
return inst
# __init__的第一个参数self是__new__魔术方法返回的结果
def __init__(self,price = 50):
print('调用__init__方法')
print(self)
self.price = price
def __call__(self,m,n):
print('调用__call__方法')
self.how_much_of_book(n) # 类里面可以随便这样调用其他方法
return self.price * n
def how_much_of_book(self,n):
print('调用how_much_of_book方法')
print(self)
return self.price * n
foo = Foo() # 通过Foo()返回一个对象
print(foo(3,2)) # 如果没有__call__(self,m,n),foo()会报错,只能通过foo.function来调用
运行结果:
调用__new__方法
调用__init__方法
调用__call__方法
调用how_much_of_book方法
100
3) 比较的魔术方法
① __cmp__(self,other) 是比较方法里面最基本的魔术方法,它实际上实现了所有的比较符号(other的时候会返回正数。通常最好的一种方式是去分别定义每一个比较符号而不是一次性将他们都定义。但是__cmp__方法是你想要实现所有的比较符号而一个保持清楚明白的一个好的方法。
② __eq__(self,other) 定义相等符号的行为,==
③ __ne__(self,other) 定义不等于符号的行为,!=
④ __It__(self,other) 定义小于符号的行为,<
⑤ __gt__(self,other) 定义大于符号的行为,>
⑥ __le__(self,other) 定义小于相等符号的行为,<=
⑦ __ge__(self,other) 定义大于相等符号的行为,>=
class A:
def __init__(self,a,b):
self.a = a
self.b = b
def __eq__(self,other): # ==号触发魔术方法
if self.a + self.b == other.a + other.b: # 需要判断两个实例是否相同,通过不同实例化对象的参数和判断,如果和相同,则这两个实例的值相同。
return True
else:
return False
def print_data(self):
print(self.a + self.b)
object1 = A(3,4)
object2 = A(3,4)
object3 = A(3,3)
object1.print_data() # 打印 object1 的实例化对象的值
print(object1 == object2) # 实例化对象的比较,触发魔术方法,比较 self.a + self.b == other.a + other.b
print(object1 == object3) # 实例化对象 object1 的值(elf.a + self.b)与实例化对象 object3 的值(other.a + other.b)不相等,
运行结果:
7
True
False
class new_list:
def __init__(self,a):
self.a = a
# 实例化方法比较两个实例大小
def gt(self,other):
list1 = list(filter(lambda x:type(x)==int,self.a))
list2 = list(filter(lambda x:type(x)==int,other.a))
if sum(list1)>sum(list2):
return True
else:
return False
# 魔术方法比较两个实例大小
def __gt__(self,other): # 如果类中没有 def __ge__()这个魔术方法,那么 print(object1 > object2) 就会报错
list1 = list(filter(lambda x:type(x)==int,self.a))
list2 = list(filter(lambda x:type(x)==int,other.a))
if sum(list1)>sum(list2):
return True
else:
return False
list1 = new_list([1,2,3,22222,'sdd','da','dasdas'])
list2 = new_list([23,'3213',232,'qdv'])
list3 = new_list([1,21,3,22,99999,'sdd','da','dasdas'])
print(list1 > list2) # 魔术方法比较
print(list1 > list3) # 魔术方法比较
print(list1.gt(list2)) # 实例化方法比较
print(list1.gt(list3)) # 实例化方法比较
print(list3 > list1 > list2) # 第一个大于符号会触动一下魔术方法,第二个大于符号也会触动一下魔术方法
print(list1 > list2 > list3)
运行结果:
True
False
True
False
True
False
class A:
def __init__(self,a,b):
self.a = a
self.b = b
def __eq__(self,other):
if self.a + self.b == other.a + other.b:
return True
else:
return False
def __str__(self): # __str__定义当被str()调用时的行为
return str(self.a + self.b)
def print_data(self):
print(self.a + self.b)
object2 = A(3,4) # 实例化对象
print(str(object2)) # 如果没有__str__魔术方法,打印的实例化对象的地址,有__str__魔术方法,该语句相当于调用 print(object2.__str__())
运行结果:
7
4) 容器类型的魔术方法
① __len__(self) 定义当被len()调用时的行为,返回容器中元素的个数
② __getitem__(self,key) 定义获取容器中指定元素的行为,相当于self[key]
③ __setitem__(self,key,value) 定义设置容器中指定元素的行为,相当于self[key] = value
④ __delitem__(self,key) 定义删除容器中指定元素的行为,相当于del self[key]
⑤ __iter__(self) 定义当迭代容器中的元素的行为
⑥ __reversed__(self) 定义当被reversed()调用时的行为
⑦ __contains__(self,item) 定义当使用成员测试运算符(in 或 not in)时的行为
class NewDataType:
def __init__(self,data):
self.data = data
def __len__(self):
return len(self.data)
object1 = NewDataType(['1','2','4',33,44]) # 实例化对象时是否传入参数,根据init来定
print(len(object1))
运行结果:
5
class NewDataType:
def __init__(self,data):
self.data = data
def __getitem__(self,key):
return self.data[key]
object1 = NewDataType(['1','2','4',33,44])
print(object1[-1])
for i in object1: # 如果没有__getitem__(self,key),无法进行遍历的
print(i)
运行结果:
44
1
2
4
33
44
class MyIter:
def __init__(self,total,start=0):
self.total = total
self.start = start
def __iter__(self):
return self
def __next__(self): # 既可以通过for循环访问也可以通过 next()方法访问
if self.start < self.total:
self.start += 1
return self.start
else:
raise StopIteration
myIter = MyIter(3)
it1 = iter(myIter) # 这个函数触发 __iter__(self):
for i in it1:
print(i)
myIter = MyIter(4) # 必须要再实例化对象一次,才能调用 it2 = iter(myIter),不能直接调用 it2 = iter(myIter)
it2 = iter(myIter) # 这个函数触发 __iter__(self):
print(next(it2))
print(next(it2))
print(next(it2))
运行结果:
1
2
3
1
2
3