什么是魔术方法

相信大家在使用python的过程中经常会看到一些双下划线开头,双下划线结尾的方法,我们把它统称为魔术方法魔术方法的特征

  • 魔术方法都是双下划线开头,双下划线结尾的方法
  • 魔术方法都是 python 内部事先定义的,是对象相关行为的底层实现方法
  • 魔术方法都是在特定的情况下自动化触发的,一般不会直接去调用

1.__new__方法

创建一个对象的时候,调用的第一个方法是__new__方法,创建并返回一个实例对象(存在于父类object中),创建的实例对象让__init__函数初始化,一般单例模式会重写该方法

class MyClass:
   def __new__(cls, *args, **kwargs):
       print("这是一个new方法")
       return super().__new__(cls, *args, **kwargs)

   def __init__(self):
       print("这是初始化方法")

my_class = MyClass()
#输出结果
#这是一个new方法
#这是初始化方法

2.__str__和__repr__方法

两个方法的作用都差不多,只不过str方法是展示给用户看的,repr通常被用于调试,两种方法如果重写的话必须要写return并且返回的是一个字符串对象

class MyClass:
  def __init__(self, name):
      self.name = name

  def __str__(self):
      print('__str__方法被触发了')
      return self.name

  def __repr__(self):
      print('__repr__方法被触发了')
      return '<MyTest.obj-{}>'.format(self.name)

my_class = MyClass("松勤")
print(my_class)
print(repr(my_class))

3.__call__方法

如果想让类创建出来的对象可以跟函数一样调用,我们就可以重写call方法 在里面写对象调用的逻辑

class MyClass:
   def __init__(self, name):
       self.name = name

   def __call__(self, *args, **kwargs):
       print("call方法被触发")

my_class = MyClass("松勤")
my_class()
#输出
#call方法被触发

4.__del__方法

当一个对象在内存中被销毁的时候自动执行,可以自动执行,我们可以用来关闭一些资源

class MyClass:
   def __init__(self, name):
       self.name = name

   def __del__(self):
       print("实例被销毁了")

my_class = MyClass("松勤")
#输出
#实例被销毁了

5.__eq__方法

我们熟知is是比较两个对象的id值是否相等,是否指向同一个内存地址,==比较的是两个对象的内容是否相等==默认会调用对象的__eq__方法,我们可以重写这个方法用来只比较内容是否相等

class MyClass:
   def __init__(self, name):
       self.name = name
   def __eq__(self, other):
       return self.__dict__ == other.__dict__

my_class = MyClass("松勤")
my_class1 = MyClass("松勤")
print(my_class==my_class1) #True
print(my_class is my_class1) #False