python 面向对象之魔术方法
前言
相信很多使用 python 的小伙伴都有一个困惑,在看一些库的源码时,发现源码中有很多 __XX__(双下划线开头,双下划线结尾)的方法。比如我们在定义类时,经常用到的初始化方法 __init__,在 python 中像 __init __ 这类双下划线开头和结尾的方法,我们把它统称为魔术方法(也有叫魔法方法和特殊方法的)。 今天就专门和大家一起来聊聊 python 中的魔术方法,首先我们来看看魔术方法有哪些特征。
魔术方法的特征:
魔术方法都是双下划线开头,双下划线结尾的方法
魔术方法都是 python 内部事先定义的,是对象相关行为的底层实现方法
魔术方法都是在特定的情况下自动化触发的,一般不会直接去调用。
接下来我们一起看看 python 中场景的一些魔术方法。
1、__new__ 方法
相信大多的程序员,都听说过 new一个对象 这句话(如下图),
在很多的编程语言中创建对象都是使用的 new 来创建的,那么在咱们 python 中呢?其实也有一个 new,它是一个魔术方法,接下来我们一起来看看。
问题:python 创建一个对象的时候,调用的第一个方法是什么?
很多小伙伴会说是 __init__,其实不然,正确答案是:__new__ 方法
问题:那么这个 __new__ 方法呢,它有什么作用?又在什么时候会调用呢?
案例:
接下来我们一起来看看下面这案例段代码: class Test(object):
def __init__(self):
print('-------init------方法')
def __new__(cls, *args, **kwargs):
print('------new方法-------')
t = Test()
print(t)
运行上面的代码,发现 __new__ 执行了,__init__ 没有执行,创建出来的对象变成了 None
为什么会出现这样的情况呢?
原因是我们重写了父类 object 中的 __new__ 方法,当我们没有自定义 __new__ 方法时,默认继承了 object 的 __new__ 方法,我们使用类创建对象时,底层会自动调用这个方法来完成对象的创建。那么我们自己定义了这个方法之后呢?方法中没没有创建对象,也没有返回对象,所以最终创建的对象打印为 None,而 __init__ 方法是一个实例方法,是创建对象之后用来初始化对象的,但是 new 方法中并没有创建出来对象,所以 __init__ 方法也没有执行。 __new__ 一般情况下我们都不会自己去重定义,只有在有特定的需求时才会去用,比如要修改或者控制类创建对象行为时,如实现单例类等等。
2、上下文管理器
相信很多小伙伴都用过 python 中的 with,也都指定可以同它来操作文件,文件会自动关闭。那么大家有没有思考过一个问题,为什么 with 打开文件为何会自动关闭?
问题: with 打开文件为何会自动关闭?
其实 with 操作文件不需要关闭的原因是,因为 with 启动的文件操作的上下文管理器协议。
什么上下文管理器协议?
所谓的上下文管理器协议,是由两个魔术方法实现的。在 python 中只要任意一个类中实现了 __enter__ 和 __exit__ 这两个方法,那么这个类就实现了上下文管理器协议,这个类的对象就可以使用 with 来进行操作。
object.__enter__(self)
使用 with 操作实现上下文管理器协议的对象时,则会自动调用这个对象的 __enter__ 方法,with 语句将该方法非返回赋值给到 as 后面的变量。
object.__exit__(self, exc_type, exc_val, exc_tb) exc_type : # 异常类型
exc_val : # 异常值
exc_tb : # 异常回溯追踪
当 with 中的代码执行完毕之后,会自动调用 __exit__ 方法退出上下文管理器。如果该上下文退出时没有异常,三个参数都将为None。如果提供了一个异常,并且该方法希望抑制该异常(即防止它被传播),它应该返回一个真值。否则,在退出此方法后,异常将被正常处理。注意__exit__()方法不应该重新抛出传递进去的异常;这是调用者的责任。
案例
手动实现操作文件的上下文管理器
class OpenFile(object):
'''手动实现文件操作的上下文'''
def __init__(self,filename,method):
#初始化打开文件
self.file = open(filename,method)
def __enter__(self):
#启动上下文时,将打开的对象返回出去
return self.file
def __exit__(self, exc_type, exc_val, exc_tb):
#退出上下文时,将文件关闭
self.file.close()
with OpenFile('python.txt','w') as f:
f.write('hello')
3、__call__ 方法
问题:python 中万物皆对象,函数也是对象,为什么函数可以调用,而其他的对象不行?
需求:如果想让类创建出来的对象,可以像函数一样被调用可以实现吗?
我们只需要再类里面定义魔术方法 __call__ 方法即可实现,__call__ 中可以定义对象调用的逻辑
class Test(object):
def __call__(self):
print('触发了call方法')
t = Test()
t()
4、__str__ 方法
问题思考:交互环境下 print 打印的内容和和直接输入变量,返回的内容不一样这是为什么?
使用 print 打印的时候触发的是 __str__ 方法,
注意点:
重写 `str,必须要记得写 return。
return 返回的必须是一个字符串对象。
代码演示:
class Test(object):
def __init__(self,name):
self.name = name
def __str__(self):
return '触发了str方法'
python 的内置函数 str
内置函数 str 转换一个对象时,触发对象对应 __str__ 的方法。
内置函数 format 处理对象是,触发对象对应 __str__ 的方法。
5、算术运算的实现
思考问题: python 中不仅数值之间能相加,字符串和列表,元祖之间也能进行,这是怎么实现的?
同类型对象之间使用 + 号的时候,实际上是触发了 __add__ 魔术方法。
小案例验证
class Test(object):
def __init__(self,name,age):
self.name = name
self.age = name
def __add__(self, other):
print('对象之间使用了+号')
return self.age+other.age
xiaoming = Test('小明',18)
laowang = Test('老王',48)
print(xiaoming+laowang)
问题二:数值之间能用-进行运算,字符型、列表元祖为什么不行?
其实也是魔术方法来实现的,python 的 int 类中实现了 __sub__ 方法,字符串,列表等数据类型没有,所有的算术运算底层都是调用相应的魔术方法来实现的。
其他算术运算符对应的魔术方法:
__add__(self, other)定义加法的行为:+
__sub__(self, other)定义减法的行为:-
__mul__(self, other)定义乘法的行为:*
__truediv__(self, other)定义真除法的行为:/
__floordiv__(self, other)定义整数除法的行为://
__mod__(self, other)定义取余算法的行为:%
关于 python 中的魔术方法就暂时给大家介绍到这里,更多的魔术方法,大家可以自行扩展学习: