python的魔法方法是如__init__()一样的,有双下划线结构的方法,在某些时候,我们需要对特定的方法进行重写,以达到需要的效果。如下,是截自其他网友的图,链接在文末。
属性访问和描述符属于魔法方法中比较常见的两类。
_getattr_(self, name)
该方法定义了你试图访问一个不存在的属性时的行为。
因此,重载该方法可以实现捕获错误拼写然后进行重定向, 或者对一些废弃的属性进行警告。
_setattr_(self, name, value)
它定义了你对属性进行赋值和修改操作时的行为。
不管对象的某个属性是否存在,它都允许你为该属性进行赋值,因此你可以为属性的值进行自定义操作。有一点需要注意,实现__setattr__时要避免"无限递归"的错误,下面的代码示例中会提到。
_delattr_(self, name)
与__setattr__很像,只是它定义的是你删除属性时的行为。
实现__delattr__是同时要避免"无限递归"的错误。
_getattribute_(self, name)
定义了你的属性被访问时的行为,相比较,_getattr__只有该属性不存在时才会起作用。
因此,在支持__getattribute__的Python版本,调用__getattr__前必定会调用_getattribute。_getattribute__同样要避免"无限递归"的错误。
需要提醒的是,最好不要尝试去实现__getattribute_,因为很少见到这种做法,而且很容易出bug。
def__setattr__(self, name, value):
self.name = value #这是错误的方式
# 每一次属性赋值时, __setattr__都会被调用,因此不断调用自身导致无限递归了。
def__setattr__(self, name, value):
self.__dict__[name] = value #这是正确的方式
def__setattr__(self, name, value):
return super().__setattr__(name, value) #或者使用super()
一个类要成为描述器,必须实现_get_,_set_,__delete__中的至少一个方法。下面简单介绍下:
_get_(self, instance, owner):在其拥有者对其读值的时候调用。
_set_(self, instance, value):在其拥有者对其进行修改值的时候调用。
_delete_(self, instance):在其拥有者对其进行删除的时候调用。
如下代码是描述符类的及魔法方法的常见应用场景。(代码粘自马老师的博客,见文末链接。)
class Celsius:
def __init__(self, value=26.6):
self.value = value
def __get__(self, instance, owner):
return self.value
def __set__(self, instance, value):
self.value = float(value)
class Fahrenheit:
def __get__(self, instance, owner):
return instance.cel * 1.8 + 32
def __set__(self, instance, value):
instance.cel = (float(value) - 32) / 1.8
class Temperature:
cel = Celsius()
fah = Fahrenheit()
temp = Temperature()
print(temp.cel) # 26.6
print(temp.fah) # 79.88
temp.cel = 30
print(temp.cel) # 30
print(temp.fah) # 86.0
temp.fah = 79.88
print(temp.cel) # 26.599999999999998
print(temp.fah) # 79.88
https://www.cnblogs.com/seablog/p/7173107.html
https://www.cnblogs.com/zhouyixian/p/11129347.html
https://blog.csdn.net/LSGO_MYP/article/details/102545056
https://blog.csdn.net/LSGO_MYP/article/details/102573292