python外篇(魔术方法)

Posted Mr.Joden

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python外篇(魔术方法)相关的知识,希望对你有一定的参考价值。

目录

归类

new和init

str和repr

是否需要实现

自实现with open


具体的官方文档学习链接放前面:  https://docs.python.org/3/reference/datamodel.html#special-method-names

归类

### 1. 对象表示相关的魔术方法:
    __str__(self):返回对象的字符串表示形式,用于将对象转化为人类可读的字符串形式。
    __repr__(self):返回对象的可打印的字符串形式,用于调试和开发,通常需要满足eval(repr(obj)) == obj。
    __format__(self, format_spec):指定对象在使用format函数进行格式化时的字符串形式。


### 2. 对象构造和初始化相关的魔术方法:
    __new__(cls, *args, **kwargs):用于创建一个新的对象,通常与类方法一起使用。
    __init__(self, *args, **kwargs):用于初始化对象的属性,通常用于初始化对象的状态和行为。
    __del__(self):用于对象的删除和垃圾回收,通常用于释放对象占用的资源。


### 3. 对象比较和排序相关的魔术方法:
    __eq__(self, other):用于判断两个对象是否相等。
    __ne__(self, other):用于判断两个对象是否不相等。
    __lt__(self, other):用于判断一个对象是否小于另一个对象。
    __le__(self, other):用于判断一个对象是否小于或等于另一个对象。
    __gt__(self, other):用于判断一个对象是否大于另一个对象。
    __ge__(self, other):用于判断一个对象是否大于或等于另一个对象。


### 4. 对象属性访问相关的魔术方法:
    __getattribute__(self, name):用于获取对象的不存在的属性时的行为,__getattr__作为备用。
    __setattr__(self, name, value):用于设置对象属性的值时的行为。
    __delattr__(self, name):用于删除对象属性时的行为。


### 5. 对象容器和迭代相关的魔术方法:
    __len__(self):用于返回对象的长度。
    __getitem__(self, key):用于获取对象的元素。
    __setitem__(self, key, value):用于设置对象的元素。
    __delitem__(self, key):用于删除对象的元素。
    __iter__(self):用于获取迭代器对象。
    __next__(self):用于获取下一个迭代器对象。


### 6. 对象运算符重载相关的魔术方法:
    __add__(self, other):用于实现加法运算。
    __sub__(self, other):用于实现减法运算。
    __mul__(self, other):用于实现乘法运算。
    __truediv__(self, other):用于实现真除法运算。
    __floordiv__(self, other):用于实现地板除法运算。
    __mod__(self, other):用于实现取模运算。
    __pow__(self, other):用于实现幂运算。
    __and__(self, other):用于实现按位与运算。
    __or__(self, other):用于实现按位或运算。
    __xor__(self, other):用于实现按位异或运算。
    __neg__(self):用于实现负号运算。
    __pos__(self):用于实现正号运算。
    __abs__(self):用于实现绝对值运算。
    __invert__(self):用于实现按位取反运算。


### 7. 上下文管理相关的魔术方法:
    __enter__(self):用于进入上下文管理器时执行的操作。
    __exit__(self, exc_type, exc_val, exc_tb):用于退出上下文管理器时执行的操作。


### 8. 其他魔术方法:
    __call__(self, *args, **kwargs):用于使对象可以像函数一样调用。
    __hash__(self):用于获取对象的哈希值。
    __bool__(self):用于判断对象的真假值。
    __instancecheck__(self, instance):用于检查对象是否是某个类的实例。
    __subclasscheck__(self, subclass):用于检查对象是否是某个类的子类。


        需要注意的是,不是所有的魔术方法都需要在自定义类中实现。根据业务需要实现所需的魔术方法即可。

new和init

### 简要:
    (1) 先执行__new__,随后执行__init__
    (2) new接受类对象,而init接受实例对象,即__new__(cls)、__init__(self)
    (3) new用于创建并返回实例对象,而init用于接受实例对象并进行初始化

    
### 详细:
    当我们创建一个类的实例对象时,首先会调用__new__()方法来创建一个实例对象,然后再调用__init__()方法来初始化这个实例对象。具体来说,__new__()方法用于创建一个空的实例对象,而__init__()方法则用于给这个实例对象添加属性和初始化值。
    换句话说,__new__()方法是在实例对象被创建之前调用的,而__init__()方法则是在实例对象被创建之后调用的。__new__()方法返回一个实例对象,这个实例对象会被传递给__init__()方法作为第一个参数(即self参数),然后在__init__()方法中对这个实例对象进行初始化操作。
    需要注意的是,__new__()方法的第一个参数是类对象(即cls参数),而__init__()方法的第一个参数是实例对象(即self参数)。另外,如果__new__()方法没有返回一个实例对象,则__init__()方法不会被调用。

### 代码示例:

class MyClass:
    # 1. __new__
    def __new__(cls, *args, **kwargs):
        # 1.1 创建实例
        instance = object.__new__(cls)
        # 1.2 返回实例
        return instance
    
    # 2. __init__
    def __init__(self,name,age):
        # 2.1 实例初始化
        self.name = name
        self.age = age

obj = MyClass('joden',20)   # new --> init

str和repr

### 简要
    (1) __str__和__repr__都接受对象实例即self
    (2) __str__用于定义对象的字符串表达形式,常用于在用户界面或日志输出等场景
    (3) __repr__用于定义对象的“代码”表示形式,常用于代码调试
    (4) 如果__str__不存在,将会用__repr__来替代__str__,repr更像是一个备用


    

### 详细:
    __str__()用于定义对象的字符串表示,_repr__()用于定义对象的“代码”表示形式。
    __str__()方法用于定义对象的字符串表示形式,即在使用print()函数或者str()函数将对象转换为字符串时,会调用该方法并返回一个字符串对象。如果一个对象没有定义__str__()方法,那么会调用__repr__()方法来代替。
    __repr__()方法用于定义对象的“代码”表示形式,即在使用内置repr()函数将对象转换为字符串时,会调用该方法并返回一个字符串对象。这个字符串对象应当是一个合法的 Python 表达式,通过执行该表达式可以重新得到该对象。如果一个对象没有定义__repr__()方法,那么会默认返回一个类似<__main__.ClassName object at 0x7f50d6199ac0>的字符串。
    注意,str和repr都与type没有任何关系。
    

### 代码示例:

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def __str__(self):
        print('__str__被调用')
        return f"self.name is self.age years old."

    def __repr__(self):
        print('__repr__被调用')
        return "Person(,)".format(self.name,self.age)

p = Person("John", 25)

# 调用print函数
print(p)  # 输出:John is 25 years old.
print(repr(p)) # 输出:Person(John,25)

是否需要实现

### 前言
    首先要明确自定的实现类都默认继承了最底层的Object类,所以也就继承了Object类中所有的魔术方法。
    在魔术方法中有些需要我们手动编码去实现以便满足我们的应用场景,当然也有些魔术方法一般不需要修改即是符合正常逻辑的,所以了解Object类中魔术方法的默认行为很重要。
    下面可以根据需求将魔术分为:需要实现与不需要实现 (当然这是一般情况)
    不同python版本的魔术方法也有一些不同,但是我们可以使用__dir__()打印出来看一下。

    
    
### 打印魔术方法(如下)
(1) 打印类的魔术方法(如下)
    print(Person.__dir__(Person))
    '__repr__', '__call__', '__getattribute__', '__setattr__', '__delattr__', '__init__', '__new__', '__subclasses__', '__prepare__', '__instancecheck__', '__subclasscheck__', '__dir__', '__sizeof__', '__basicsize__', '__itemsize__', '__flags__', '__weakrefoffset__', '__base__', '__dictoffset__', '__mro__', '__name__', '__qualname__', '__bases__', '__module__', '__abstractmethods__', '__dict__', '__doc__', '__text_signature__', '__hash__', '__str__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__class__'   
    可以看到,有__getattribute__而不是__getattr__(备用)
    
(2) 打印对象的魔术方法(如下)
    print(p1.__dir__())
    '__module__', '__init__', '__str__', '__repr__', '__dict__', '__doc__', '__hash__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__'
    可以看到对比类中的少了很多,这些都是内部自动从对象中已经继承的魔术方法
    
    
    
### 类中好用的魔术方法
    __bases__:返回一个tuple包含了当前类的所有直接父类(一层)
    __base__:返回一个class为当前类的首个直接父类(一层首个)
    __subclasses__():返回一个list包含了当前类的所有直接子类(一层)
    __mro__:返回一个tuple包含了当前类的所有继承关系类(多层)
    mro():返回一个list包含了当前类的所有继承关系类(多层)
    __dict__:返回一个字典包含了当前类中所有的静态属性、实例属性、方法
    
    

### 不需要实现:
    __dir__(self):获取对象的属性和方法的列表,默认即可
    __doc__(self):获取类中的文档字符串(三引号包裹的字符串)且不存在时返回None,默认即可
    __class__(self):获取对象的类,默认即可
    __module__(self):获取类定义所在默认的模块名称,默认即可
    __dict__:获取实例对象的属性字典(不包含方法),默认即可
    __reduce__:对象序列化和反序列化被调用的方法,默认即可
    __new__(cls):创建实例并返回嘛又也不是初始化,默认即可
    __del__(self):销毁实例,垃圾回收机制已经帮我们做了,默认即可
    __getattribute__(self,name):获取属性且不存在时抛出AttributeError,一般默认
    __setattr__(self,name,value):为对象设置绑定的属性,默认即可
    __delattr__(self,name):删除属性不存在时抛出AttributeError,默认即可
    __sizeof__(self):获取对象占用的内存字节数,默认即可
    __format__(self,format_spec):获取对象的格式化字符串表示会抛出TypeError,默认即可


### 需要实现:
    __init__(self,*args,**kwargs):实例化创建对象,需要去实现
    __str__(self):对象的字符串表现形式,需要我们去实现
    __repr__(self):对象的“代码”表现形式,需要我们去实现
    __call__(self):回调方法,默认会在对象被调用的时候执行(即在init方法之后执行),需要我们去显式的定义
    __eq__(self,obj):对象比较时调用,默认比较内存id是否相等于,需要实现
    __ne__(self,obj):对象比较时调用,默认比较内存id是否不相等于,需要实现
    __lt__(self,obj):对象比较时调用,默认比较内存id是否小于,需要实现
    __le__(self,obj):对象比较时调用,默认比较内存id是否小于等于,需要实现
    __gt__(self,obj):对象比较时调用,默认比较内存id是否大于,需要实现
    __ge__(self,obj):对象比较时调用,默认比较内存id是否大于等于,需要实现
    __enter__(self):使用with语法时就会调用它,并且__enter__(self)的返回值会被赋值给as后面的变量,可用于实现上下文管理类(像我们使用的with open就是该原理)
    __exit(self,exc_type,exc_value,traceback):和__enter__(self)结合使用,当with下的代码块被中完成后__exit__()将会被执行,通常我们在这里进行释放那个资源、清理环境等操作(像我们使用的with open就是该原理)

自实现with open

class MyOpen:
	def __init__(self,filename,mode):
		self.filename = filename
		self.mode = mode
		
	def __enter__(self):
        # 在这里创建一个文件流
        self.file = open(self.filename,self.mode)
        return self.file
			
	def __exit__(exc_type,exc_value,exc_tb):
        # 在这里把文件流关闭
        self.file.close()
	
    
with MyOpen('test.txt','w') as f:	# with会自动执行__enter__,并且有 f = self.file
	f.write("Hello Joden")	# 执行完会自动执行__exit__

Python 类的魔术方法

Python中类的魔术方法

  在Python中以两个下划线开头的方法,__init__、__str__、__doc__、__new__等,被称为"魔术方法"(Magic methods)。魔术方法在类或对象的某些事件出发后会自动执行,如果希望根据自己的程序定制自己特殊功能的类,那么就需要对这些方法进行重写。
  注意:Python 将所有以 __(两个下划线)开头的类方法保留为魔术方法。所以在定义类方法时,除了上述魔术方法,建议不要以 __ 为前缀。

Python提供的魔术方法

  魔术方法这里按照不同的类别有如下分类:

魔法方法
含义
 
基本的魔法方法
__new__(cls[, ...]) 1. __new__ 是在一个对象实例化的时候所调用的第一个方法
2. 它的第一个参数是这个类,其他的参数是用来直接传递给 __init__ 方法
3. __new__ 决定是否要使用该 __init__ 方法,因为 __new__ 可以调用其他类的构造方法或者直接返回别的实例对象来作为本类的实例,如果 __new__ 没有返回实例对象,则 __init__ 不会被调用
4. __new__ 主要是用于继承一个不可变的类型比如一个 tuple 或者 string
__init__(self[, ...]) 构造器,当一个实例被创建的时候调用的初始化方法
__del__(self) 析构器,当一个实例被销毁的时候调用的方法
__call__(self[, args...]) 允许一个类的实例像函数一样被调用:x(a, b) 调用 x.__call__(a, b)
__len__(self) 定义当被 len() 调用时的行为
__repr__(self) 定义当被 repr() 调用或者直接执行对象时的行为
__str__(self) 定义当被 str() 调用或者打印对象时的行为
__bytes__(self) 定义当被 bytes() 调用时的行为
__hash__(self) 定义当被 hash() 调用时的行为
__bool__(self) 定义当被 bool() 调用时的行为,应该返回 True 或 False
__format__(self, format_spec) 定义当被 format() 调用时的行为
 
有关属性
__getattr__(self, name) 定义当用户试图获取一个不存在的属性时的行为
__getattribute__(self, name) 定义当该类的属性被访问时的行为
__setattr__(self, name, value) 定义当一个属性被设置时的行为
__delattr__(self, name) 定义当一个属性被删除时的行为
__dir__(self) 定义当 dir() 被调用时的行为
__get__(self, instance, owner) 定义当描述符的值被取得时的行为
__set__(self, instance, value) 定义当描述符的值被改变时的行为
__delete__(self, instance) 定义当描述符的值被删除时的行为
 
比较操作符
__lt__(self, other) 定义小于号的行为:x < y 调用 x.__lt__(y)
__le__(self, other) 定义小于等于号的行为:x <= y 调用 x.__le__(y)
__eq__(self, other) 定义等于号的行为:x == y 调用 x.__eq__(y)
__ne__(self, other) 定义不等号的行为:x != y 调用 x.__ne__(y)
__gt__(self, other) 定义大于号的行为:x > y 调用 x.__gt__(y)
__ge__(self, other) 定义大于等于号的行为:x >= y 调用 x.__ge__(y)
 
算数运算符
__add__(self, other) 定义加法的行为:+
__sub__(self, other) 定义减法的行为:-
__mul__(self, other) 定义乘法的行为:*
__truediv__(self, other) 定义真除法的行为:/
__floordiv__(self, other) 定义整数除法的行为://
__mod__(self, other) 定义取模算法的行为:%
__divmod__(self, other) 定义当被 divmod() 调用时的行为
__pow__(self, other[, modulo]) 定义当被 power() 调用或 ** 运算时的行为
__lshift__(self, other) 定义按位左移位的行为:<<
__rshift__(self, other) 定义按位右移位的行为:>>
__and__(self, other) 定义按位与操作的行为:&
__xor__(self, other) 定义按位异或操作的行为:^
__or__(self, other) 定义按位或操作的行为:|
 
反运算
__radd__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rsub__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rmul__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rtruediv__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rfloordiv__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rmod__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rdivmod__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rpow__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rlshift__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rrshift__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rand__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__rxor__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
__ror__(self, other) (与上方相同,当左操作数不支持相应的操作时被调用)
 
增量赋值运算
__iadd__(self, other) 定义赋值加法的行为:+=
__isub__(self, other) 定义赋值减法的行为:-=
__imul__(self, other) 定义赋值乘法的行为:*=
__itruediv__(self, other) 定义赋值真除法的行为:/=
__ifloordiv__(self, other) 定义赋值整数除法的行为://=
__imod__(self, other) 定义赋值取模算法的行为:%=
__ipow__(self, other[, modulo]) 定义赋值幂运算的行为:**=
__ilshift__(self, other) 定义赋值按位左移位的行为:<<=
__irshift__(self, other) 定义赋值按位右移位的行为:>>=
__iand__(self, other) 定义赋值按位与操作的行为:&=
__ixor__(self, other) 定义赋值按位异或操作的行为:^=
__ior__(self, other) 定义赋值按位或操作的行为:|=
 
一元操作符
__pos__(self) 定义正号的行为:+x
__neg__(self) 定义负号的行为:-x
__abs__(self) 定义当被 abs() 调用时的行为
__invert__(self) 定义按位求反的行为:~x
 
类型转换
__complex__(self) 定义当被 complex() 调用时的行为(需要返回恰当的值)
__int__(self) 定义当被 int() 调用时的行为(需要返回恰当的值)
__float__(self) 定义当被 float() 调用时的行为(需要返回恰当的值)
__round__(self[, n]) 定义当被 round() 调用时的行为(需要返回恰当的值)
__index__(self) 1. 当对象是被应用在切片表达式中时,实现整形强制转换
2. 如果你定义了一个可能在切片时用到的定制的数值型,你应该定义 __index__
3. 如果 __index__ 被定义,则 __int__ 也需要被定义,且返回相同的值
 
上下文管理(with 语句)
__enter__(self) 1. 定义当使用 with 语句时的初始化行为
2. __enter__ 的返回值被 with 语句的目标或者 as 后的名字绑定
__exit__(self, exc_type, exc_value, traceback) 1. 定义当一个代码块被执行或者终止后上下文管理器应该做什么
2. 一般被用来处理异常,清除工作或者做一些代码块执行完毕之后的日常工作
 
容器类型
__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)时的行为

Python的魔术方法举例

这里仅针对常用的魔术方法进行举例,便于理解及灵活运用,部分魔术方法已经在:这篇博客中举例说明

小技巧:如果在需要返回对象的魔术方法里面不知道如何返回,可以调用super函数来执行父类的相同方法。

注意:魔法方法必须要使用return进行返回。

基本的魔法方法

__repr__(self):直接执行对象时执行的方法

>>>
>>> class A:
    def __repr__(self):
        return \'__repr__\'
     
>>> a = A()
>>> a
__repr__

__bool__(self):判断对象的bool值时执行的方法,返回值只能是bool类型

>>>
>>> class C:
    def __bool__(self):
        return False
     
>>> c = C()
>>> bool(c)
False
>>>

  

属性相关的魔术方法

__getattr__(self,name):获取一个不存在的属性时执行的方法。

__setattr__(self,name,value):设置一个属性时执行的方法。

__delattr__(self,name):删除一个属性时执行的方法。

__getattribute__(self,name):当该类的属性被访问时执行的方法。

>>> class A:
    def __getattr__(self,name):
        print(\'__getattr__\')
    def __setattr__(self,name,value):
        print(\'__setattr__\')
        super().__setattr__(name,value)
    def __delattr__(self,name):
        print(\'__delattr__\')
        return super().__delattr__(name)
    def __getattribute__(self,name):
        print(\'__getattribute__\')
        return super().__getattribute__(name)
 
     
>>> a = A()
>>> a.name
__getattribute__         
__getattr__
>>> a.name = \'daxin\'
__setattr__
>>> a.name
__getattribute__
\'daxin\'
>>> del a.name
__delattr__
>>>

  

注意

  1、__getattribute__,先于__getattr__访问

  2、__getattr__,没有在父类中进行定义,所以不能继承

  3、__setattr__,为设置属性,所以无需return

  3、在__setattr__时,不能直接使用self.name = value ,会造成死循环,因为在执行self.name = value的时候又会调用本身,无限循环下去

小技巧

  在__setattr__的时候,除了使用super的方式设置变量和值以外,还可以使用__dict__来设置。(但是建议使用super的方法)

>>> class A:
    def __getattr__(self,name):
        print(\'__getattr__\')
    def __setattr__(self,name,value):
        print(\'__setattr__\')
        self.__dict__[name] = value
    def __delattr__(self,name):
        print(\'__delattr__\')
        return super().__delattr__(name)
    def __getattribute__(self,name):
        print(\'__getattribute__\')
        return super().__getattribute__(name)
 
 
>>> a = A()
>>> a.name
__getattribute__
__getattr__
>>> a.name = \'daxin\'
__setattr__
__getattribute__
>>> a.name
__getattribute__
\'daxin\'
>>>

  

属性相关之property

我们知道使用property可以把,类的某些方法当作属性进行访问,赋值,修改的,而Property内部也是通过类的魔术方法实现的。

下面的三个属性,主要用于在被当作其他的类的属性时使用的。

__set__(self,instance,value):当作一个描述符(属性)被赋值时执行的方法

__get__(self,instance,owner):当作一个描述符(属性)被获取时执行的方法

__delete__(self,instance):当作一个描述符(属性)被删除时执行的方法

 # 当作一个描述符被访问的含义就是:
 # 一个函数在实例化的时候调用了其他类的对象
  
 >>>
 >>> class A:
      pass
  
 >>> class B:
      def __init__(self):
         self.name = A()     # 把A类的对象在B实例化的时候当作描述符(属性)进行赋值
     
 >>> 

 

>>> class A:
    def __get__(self,instance,owner):
        print(\'__get__\',\'self:\',self,\'instance:\',instance,\'owner:\',owner)
     
    def __set__(self,instance,value):
        print(\'__set__\',\'self:\',self,\'instance:\',instance,\'value:\',value)
 
    def __delete__(self,instance):
        print(\'__delete__\',\'self:\',self,\'instance:\',instance)
 
>>> class B:
    name = A()
 
>>>
>>> b = B()
>>> b.name
__get__ self: <__main__.A object at 0x1134247f0> instance: <__main__.B object at 0x113411d68> owner: <class \'__main__.B\'>
>>> b.name = \'abc\'
__set__ self: <__main__.A object at 0x1134247f0> instance: <__main__.B object at 0x113411d68> value: abc
>>> del b.name
__delete__ self: <__main__.A object at 0x1134247f0> instance: <__main__.B object at 0x113411d68>
>>>

  

扩展

由于不知道魔术方法传递的参数都是什么东西,所以在上面的例子打印了一下:

  • self:表示的是当前类(拥有__set__,__get__等方法的类本身)
  • instance:表示的是调用此类的类的实例化对象
  • owner:表示调用此类的类
  • value:表示要设置的值

根据这三个方法,那么我们完全可以定义自己的property属性

class Myproperty:   # 定义自己的Property
 
    def __init__(self,mget=None,mset=None,mdel=None):   # 绑定三个修改方法
        self.mget = mget
        self.mset = mset
        self.mdel = mdel
 
    def __get__(self,instance,owner):
        return self.mget(instance)       
 
    def __set__(self,instance,value):
        return self.mset(instance,value)
 
    def __delete__(self,instance):
        return self.mdel(instance)
 
     
class A:
    def __init__(self):
        self._x = None
 
    def setX(self,value):
        self._x = value
 
    def getX(self):
        try:
            if self._x:
                return self._x
        except AttributeError as e:
            print(\'Attribute is not exist\')
 
    def delX(self):
        del self._x
 
    name = Myproperty(getX,setX,delX)
   
 
>>> a = A()
>>> a.name
>>> a.name = \'daxin\'
>>> a.name
\'daxin\'
>>> del a.name
>>> a.name
Attribute is not exist
>>>

  

PS:关于为什么在Myproperty中传递instance,那么请注意我们的setX,getX,默认有一个self参数,使用自定义Property的话,那么我们必须手动传递,在Myproperty类中instance表示对象自己,所以传递instance给setX/getX,其实等同于self。

运算符相关魔术方法  

__add__:在执行加法时执行的方法

>>> class A(int):
    def __add__(self,other):
        print(\'__add__\')
        return super().__sub__(other)     # 这里改写了 add方法,调用了父类的减法
 
>>> a = A(8)
>>> a + 6
__add__
2
>>>

  

__radd__:当左边的对象没有__add__方法时,执行右边对象的__radd__方法

>>> class A:
    pass
 
>>> class B(int):
 
    def __radd__(self,other):
        return \'I am B,because right object not have __add__\'
 
     
>>>
>>> a = A()
>>> b = B(12)
>>> a + b
\'I am B,because right object not have __add__\'
>>>

  

容器类

在Python中,列表、元组、字典和集合等等,这些都可以称为容器。

__getitem__:获取容器中的元素,比如mylist[1]

__setitem__:设置容器中的元素,比如mylist[2] = \'hello\'

__delitem__:删除容器中的某个元素,比如 del mylist[2]

# 定义一个不可变的列表类型
# 记录元素被访问的次数
 
class Mylist:
 
    def __init__(self,*args):
        self.value = [ x for x in args ]    # 列表生成式
        self.count = {}.fromkeys(range(len(self.value)),0)    # 定义字典用于按照索引统计访问次数
 
    def __getitem__(self,key):
        self.count[key] += 1
        return self.value[key]
 
    def __repr__(self):
        return str(self.value)
 
    __str__ = __repr__
 
    
>>> a = Mylist(1,2,3,45)
>>> a
[1, 2, 3,45]
>>> a.count
{0: 0, 1: 0, 2: 0, 3: 0}
>>> a[0]
1
>>> a.count
{0: 1, 1: 0, 2: 0, 3: 0}
>>> a[2]
3
>>> a.count
{0: 1, 1: 0, 2: 1, 3: 0}
>>> a[1] = 123

  

小练习

1、写一个矩形的类,那么这个类默认有长和宽两个参数,当传入一个叫square的属性时,值就等于边长,由于是正方形那么长和宽就等于边长。

class Rectangle(object):
 
    # init object
    def __init__(self,width,height):
        self.width = width
        self.height = height
 
 
    def getacreage(self):
        return self.width * self.height
 
    def __setattr__(self,name,value):
        if name == \'square\':
            self.width = value
            self.height = value
 
        else:
            super().__setattr__(name,value)
            

  

2、完成如下要求:

  - 定制一个计时器类

  - start和stop方法代表启动和停止计时

  - 假设计时器对象t1,定制t1对象直接执行和打印时的输出信息

  - 当计时器未启动或已经停止计时,调用stop方法会给予温馨提示

  - 两个计时器对象可以进行相加: t1 + t2

import time
 
class Mytimer():
 
    # 初始化
    def __init__(self):
        self.prompt = \'Timer is not start\'    # 提示信息
        self.start_time = None  
        self.stop_time = None
     
    # 启动定时器
    def start(self):
        if self.start_time:
            print(\'Please Use Stop() to stop Timer\')   # 属性存在,表示已经启动计时
        else:
            self.start_time = time.time()
            print(\'The start of Timer\')
 
    # 关闭定时器
    def stop(self):
        if self.start_time:
            self.stop_time = time.time()
            print(\'The stop of Timer\')
            self._calc()
        else:
         print(\'Please Use Start() to start Timer\')   # 没有执行start(),检测不到属性,进行提示
 
    # 计算耗时
    def _calc(self):
        self.prompt = \'Total time : \'
        self.end_time = self.stop_time - self.start_time
        self.prompt += str(\'{:.2f}\'.format(self.end_time))
        self.start_time = None
        self.stop_time = None
 
    # 直接执行对象时,打印提示信息
    def __repr__(self):
        return self.prompt
 
    # 打印对象和执行对象相同
    __str__ = __repr__
 
    # 计时器相加
    def __add__(self,other):
        self.prompt = \'All Time : \'
        add_time = self.end_time + other.end_time
        self.prompt += str(\'{:.2f}\'.format(add_time))
        return self.prompt

  

3、完成如下需求

  - 定义一个温度类,然后定义两个描述符类用于描述摄氏度和华氏度两个属性。

  - 两个属性会自定进行转换,也就是说可以给摄氏度这个属性赋值,然后打印的华氏度属性是自动转换后的结果。

class Celsius:
 
    def __init__(self,value = 37):
        self.value = value
 
    def __set__(self,instance,value):
        self.value = value
 
    def __get__(self,instance,owner):
        return self.value
 
 
class Fahrenheit:
 
    def __set__(self,instance,value):
         instance.cel = (float(value) - 32 )  / 1.8
          
    def __get__(self,instance,owner):
        return instance.cel * 1.8  + 32   # 通过instance实例访问,实例化后对象的cel属性
 
class Temperature:
    cel = Celsius()
    fah = Fahrenheit()
 
     
>>> a = Temperature()
>>> a.cel
37
>>> a.fah
98.60000000000001
>>> a.fah = 100
>>> a.cel
37.77777777777778
>>>

 PS:由于instance表示的是实例化的对象本身(a),所以这里使用instance,指代a,来访问设置的cel属性,来进行换算。

以上是关于python外篇(魔术方法)的主要内容,如果未能解决你的问题,请参考以下文章

怎么让你的Python代码更优雅!

Python100天学习笔记番外篇 Python面试面经笔试题目汇总

Python100天学习笔记番外篇 Python面试面经笔试题目汇总

Python 类的魔术方法

Python爬虫番外篇之关于登录

python魔术方法