面向对象-常用魔术方法

魔术方法的含义

  • 魔法函数(Magic Methods),是Python的一种高级语法,允许在类中自定义函数(函数名格式一般为__xx__),并绑定到类的特殊方法中。
  • 魔术方法会在特定的情况下自动调用。

常用魔术方法

__new__方法

  • 实例化对象,向内存申请空间,返回对象的地址。
  • __new__至少有一个cls(class)的参数,代表当前类。
  • __new__必须有返回值,返回实例化出来的对象,可以return父类:return object.new (cls) 或 return super(). new(cls)

__init__方法

  • 初始化一个类,在创建实例对象过程中为其初始化赋值时调用。
class Person:
    def __init__(self, name):
        print("-----------init------------")
    def __new__(cls, *args, **kwargs):
        print("-----------new------------")
        result = super().__new__(cls)
        print(result)
        return result
p1 = Person("Jim")
print(p1)

result:
-----------new------------
<__main__.Person object at 0x7fcf980ce898>
-----------init------------
<__main__.Person object at 0x7fcf980ce898>

在这里插入图片描述

__new__和__init__的区别

  • __init__是初始化方法,需要self参数,代表__new__返回的实例,创建对象后立刻被默认调用,可接收参数,__init__不需要返回值。
  • __new__需要cls(class)的参数,代表当前类,此参数在实例化时由python解析器自动识别。
  • __new__必须要有返回值,返回实例化出来的实例,可以return父类:return object.new (cls) 或 return super(). new(cls) 。
  • 如果__new__创建的是当前实例,那么它会自动调用__init__ 函数,如果是其他类的类名,那么它创建返回的就是其他类的实例,不会调用当前类的__init__函数,也不会调用其他类的__init__函数。

__call__方法

  • __call__方法的功能类似于在类中重载 () 运算符,使得类实例对象可以像调用普通函数那样,以“对象名()”的形式使用。
  • 可以传参数,至少有一个self。
  • 将对象当做函数调用对象名()时,触发此方法。
class Student:
    def __init__(self, name, age):
        self.name = name
        self.age = age
    def __call__(self, hobby):
        print(f"{self.name} likes {hobby}.")
        
s1 = Student("GaoZhan", 20)
print(s1)
s1("swimming")

result:
<__main__.Student object at 0x7f78ae9c47b8>
GaoZhan likes swimming.

__del__方法

  • 当一个对象或的引用次数为0时,会自动调用__del__方法。
  • 单独执行del 对象不一定会执行__del__方法,还要看是否该对象有其它的引用。

对象赋值

  • 把一个对象赋值给一个变量,二者指向同一个地址。
    在这里插入图片描述

查看对象的引用次数

  • 使用sys模块中的getrefcount方法。
import sys
class Animal(object):
    def __init__(self, name):
        self.__name = name
    def __del__(self):
        print("__del__方法被调用")
        print("%s被销毁..." % self.__name)

dog = Animal("癞皮狗")

wolf = Animal("灰太狼")
wolf2 = wolf
wolf3 = wolf
print(sys.getrefcount(dog))         # 调用getrefcount方法时也会引用一次
print(sys.getrefcount(wolf))         # 调用getrefcount方法时也会引用一次
print("------------1---------------")
del dog
print("------------2---------------")
print("------------3---------------")
del wolf
print("------------4---------------")
del wolf2
print("------------5---------------")
del wolf3

result:
2
4
------------1---------------
__del__方法被调用
癞皮狗被销毁...
------------2---------------
------------3---------------
------------4---------------
------------5---------------
__del__方法被调用
灰太狼被销毁...

__str__方法

  • __str__方法用于返回对象的描述信息,如果不使用__str__方法,直接print,或者return,返回的是对象的内存地址。
  • 如果在__str__中定义了描述信息,print或者return时,返回的就不是内存地址,显示更友好,实现了类到字符串的转化。
class Student1:
    def __init__(self, name, age):
        self.name = name
        self.age = age
        
class Student2:
    def __init__(self, name, age):
        self.name = name
        self.age = age
    def __str__(self):
        return "%s的年龄是:%d" % (self.name, self.age)

zs1 = Student1("张三", 20)
print(zs1)
zs2 = Student2("张三", 20)
print(zs2)
result:
<__main__.Student1 object at 0x7fab6a76db00>
张三的年龄是:20

__repr__方法

  • 类似于__str__方法
  • 面向开发者,想要直接输出对象(如在控制台直接敲变量名)和使用 print 输入对象都显示的是友好提示。
class Test:
    def __repr__(self):
        return "------repr-------"
t = Test()
print(t)
print(t.__repr__())
print(id(t))
print(id(t.__repr__()))

result:
------repr-------
------repr-------
2862916603072
2862918614160

__repr__和__str__方法区别

  • 这两个成员的作用类似。
  • 当同时存在__str__和__repr__时,那么当打印实例对象的时候,python底层会优先执行实例对象.str()
# 直接输出对象时,带有__repr__的会输出__repr__的内容。而只带有__str__的还是默认输出。
class Test:
...     def __str__(self):
...         return "------str-------"
... t = Test()
t
<__main__.Test object at 0x00000252B0454908>

class Test:
...     def __repr__(self):
...         return "------repr-------"
... t = Test()
t
------repr-------
# 同时存在时, print优先__str__的内容。
class Test:
    def __repr__(self):
        return "------repr-------"
    def __str__(self):
        return "------str-------"
t = Test()
print(t)

result:
------str-------
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 魔术方法Python的特殊方法,它们以双下划线开头和结尾,例如__init__、__str__、__add__等。这些方法可以在类的实例化、运算符重载、属性访问等方面提供特殊的行为。 __init__方法是一个特殊的构造函数,用于初始化类的实例。__str__方法用于返回对象的字符串表示形式,可以通过print函数输出。__add__方法用于重载加法运算符,可以实现自定义的加法操作。其他常用魔术方法还包括__eq__、__lt__、__gt__等,用于比较运算符的重载。 学习魔术方法可以让我们更好地理解Python面向对象编程的特性,提高代码的可读性和可维护性。 ### 回答2: 魔术方法Python最有趣且也是最强大的概念之一。魔术方法(也称为特殊方法或双下划线方法)是一些特殊的方法,它们以双下划线(__)开头和结尾,并具有特定的名称。 这些特殊方法可以为我们提供许多有用的功能,例如重载操作符,处理类的属性,实现自定义迭代器,使用描述符等。 下面是一些常见的魔术方法: __init__:这是最常见的魔术方法。当创建一个实例时,它会被自动调用。它用于初始化对象的属性。 __str__:当你想要将一个对象转换成字符串时,这个方法会被调用。如果你不指定__str__方法,Python默认会使用对象的类名和内存地址来表示对象。 __repr__:这个方法和__str__方法类似,也是用于将对象转换成字符串。但是__repr__方法在调试时有很大的作用,因为它返回的字符串可以用来唯一地标识对象。 __len__:这个方法可以返回对象的长度。例如,如果你想获取一个字符串的长度,你可以使用len("hello"),在底层,它实际上是调用了字符串对象的__len__方法。 __getattr__和__setattr__:这些方法允许你动态地获取和设置对象的属性。当你访问一个不存在的属性时,__getattr__方法会被调用。当你设置一个属性时,__setattr__方法会被调用。 __call__:这个方法允许你将对象作为函数调用。当你调用一个对象时,Python实际上是在调用对象的__call__方法。 除了上面列举的方法,还有许多其他的魔术方法,例如__cmp__,__hash__,__iter__等等。学习这些魔术方法将使你能够更好地理解Python面向对象编程模型。 总之,学习和理解魔术方法Python面向对象编程的一个关键概念,因为它们可以帮助你实现更加灵活和强大的代码。如果你想成为一名Python高手,那么深入学习魔术方法是不可避免的。 ### 回答3: Python的“魔术方法”指的是每个类定义的特殊方法,它们以双下划线(__)开头和结尾,并且有着特定的用途。通过使用这些魔法方法,我们可以自定义类的行为,并为程序提供更高级别的功能。 以下是Python常用的一些魔术方法: 1. __init__:这是最常用魔术方法之一,它用于初始化一个类的对象,以及定义类的属性和方法。 2. __str__:此方法用于返回对象的字符串表示形式,类似于Java的toString()方法。 3. __repr__:与__str__类似,但是返回的是对象的“官方”字符串表示形式,通常用于调试和开发。 4. __getattr__:当试图访问一个不存在的属性时,此方法被调用。 5. __setattr__:当尝试设置类的属性时,此方法被调用。 6. __delattr__:当尝试删除类的属性时,此方法被调用。 7. __call__:将对象作为函数调用时,此方法被调用。 8. __len__:返回对象的长度。 9. __getitem__:允许通过索引访问对象的元素。 10. __setitem__:允许通过索引设置对象的元素。 11. __delitem__:允许通过索引删除对象的元素。 通过了解和使用这些魔术方法,我们可以编写出更高效、更灵活、更具可读性的Python代码,并且实现类似于内置类型一样的功能。例如,我们可以实现一个自定义列表,类似于Python的list类型,然后使用上述魔术方法来访问、设置和删除元素。同时,我们还可以自定义变量和函数的行为,使我们的Python代码变得更具有表现力和弹性。 总之,了解和掌握Python魔术方法Python编程必不可少的一部分,对于理解和编写实际应用程序非常有价值。在实践,我们可以根据实际情况选择恰当的魔术方法,从而创建更灵活、更高效的Python类。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值