Python基础学习简记--魔法方法(Day15)

Python基础学习简记–魔法方法(Day15)

python的魔法方法是如__init__()一样的,有双下划线结构的方法,在某些时候,我们需要对特定的方法进行重写,以达到需要的效果。如下,是截自其他网友的图,链接在文末。
Python基础学习简记--魔法方法(Day15)_第1张图片
属性访问和描述符属于魔法方法中比较常见的两类。

1、属性访问

_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()

2、描述符

一个类要成为描述器,必须实现_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

3、参考文献

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

你可能感兴趣的:(python)