鱼C论坛

 找回密码
 立即注册
查看: 2427|回复: 4

[学习笔记] 【第041讲心得】【认识__new__构造函数】

[复制链接]
发表于 2018-12-28 15:28:20 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
本帖最后由 heidern0612 于 2018-12-28 17:53 编辑

写心得的过程都是自我思考的过程,借鉴了很多论坛前辈和互联网大佬的经验,仓促间难免有疏漏,如有错误,恳请指出,不胜感激。

一、 回顾__init__方法:

之前学过__init__方法,大概很多同学都很熟悉了,__init__ 方法通常用在初始化一个类实例的时候。


例如:
class Person(object):
    """功夫巨星"""

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

    def __str__(self):
        return f'<人物是{self.name},年龄为{self.age}>'

if __name__ == '__main__':
    person = Person('Tony Jaa', 24)
    print (person)

这样便是__init__最普通的用法了。

但__init__其实不是实例化一个类的时候第一个被调用 的方法。

当使用 Persion(name, age) 这样的表达式来实例化一个类时,最先被调用的方法 其实是 __new__ 方法。



二、__new__ 方法是咩?(原谅我调皮放荡不羁爱自由

当你实例化一个对象的时候,首先就会执行__new__ 方法里面的方法。

__new__方法在类定义中不是必须写的,如果没定义,默认会调用object.__new__去创建一个对象。

__new__方法接受的参数虽然也是和__init__一样,但__init__是在类实例创建之后调用,而 __new__方法正是创建这个类实例的方法。
class Person(object):
    """功夫巨星"""
    
    def __new__(cls, name, age):
        print ('__new__在实例化之前被调用。')
        return object.__new__(cls)

    def __init__(self, name, age):
        print("__init__在实例化后被调用。")
        self.name = name
        self.age = age

    def __str__(self):
        return f'<人物是{self.name},年龄为{self.age}>'



if __name__ == '__main__':
    
    person = Person('Tony Jaa', 24)
    
    print (person)


通过运行这段代码,我们可以看到,__new__方法的调用是发生在__init__之前的。



其实当 你实例化一个类的时候,python具体的执行逻辑是这样的:

1.person = Person(name, age)实例化;

2.首先执行使用name和age参数来执行Person类的__new__方法,这个__new__方法会 返回Person类的一个实例;

     *需要注意的是,这里__new__方法必须有一个返回值,返回实例化出来的实例。

      可以return父类__new__()出来的实例,也可以直接将object的__new__()出来的实例返回。

3.然后利用这个实例来调用类的__init__方法,上一步里面__new__产生的实例也就是 __init__里面的的self;



所以,__init__ 和 __new__ 最主要的区别在于:

1.__init__ 通常用于初始化一个新实例,控制这个初始化的过程,比如添加一些属性, 做一些额外的操作,发生在类实例被创建完以后。它是实例级别的方法。

2.__new__ 通常用于控制生成一个新实例的过程。它是类级别的方法。



但是说了这么多,__new__最通常的用法是什么呢,我们什么时候需要__new__?


三、__new__ 的作用

依照Python官方文档的说法,__new__方法主要是当你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。

P.S:还有就是实现自定义的metaclass(元类)。

首先我们来看一下第一个功能,具体我们可以用int来作为一个栗子:

假如我们需要一个永远都是正数的整数类型,通过继承int,我们可能会写出这样的代码。
class PositiveInteger(int):
    def __init__(self, value):
        int.__init__(abs(value))
        
i = PositiveInteger(-3)

print (i)

但运行后会发现,结果根本不是我们想的那样,我们任然得到了-3。

这是因为对于int这种 不可变的对象,我们只有重载它的__new__方法才能起到自定义的作用。

这是修改后的代码:
class PositiveInteger(int):
    def __new__(cls, value):
        return int.__new__(cls, abs(value))
        
i = PositiveInteger(-3)

print (i)

通过重载__new__方法,我们实现了需要的功能。




总结以上:

__new__方法通常用来控制的实例产生的过程,它可以对这个类做一些配置或者处理。

__init__方法主要是用来初始化一些实例的参数,比如添加一些属性,或者做一些其他的事情,这个初始化过程发生在实例被创建以后。



__new__用于对象的创建,是一个静态方法,第一个参数是cls。(想想也是,不可能是self,对象还没创建,哪来的self)

__init__ 用于对象的初始化, 是一个实例方法,第一个参数是self。


先有创建,才有初始化。

即先__new__,而后__init__。






评分

参与人数 2荣誉 +7 鱼币 +8 贡献 +6 收起 理由
weixing123 + 5 + 5 + 3 鱼C有你更精彩^_^
小甲鱼 + 2 + 3 + 3 鱼C有你更精彩^_^

查看全部评分

本帖被以下淘专辑推荐:

想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2019-1-28 13:26:10 | 显示全部楼层
0.类方法中名字前后以__包围的方法名
1.__new__()
2.需要传入参数的时候
3.__init__()方法返回值为None
4.你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径
5.对象被垃圾回收机制启用的时候自动调用


1.class C2F():
    def __init__(self,num):
        self.num = num
    def __repr__(self):
        print(self.num*1.8+32)

2.def str2int(value):
    num = 0
    for i in value:
        num += ord(i)
    return num

class Nint(int):
    def __new__(cls,value):
        if isinstance(value,str):
            return int.__new__(cls, str2int(value))
        else:
            return int.__new__(cls,value)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-4-6 11:17:35 | 显示全部楼层
__new__用来做单例用的,楼主说明的很不错。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-4-7 10:42:59 | 显示全部楼层
什么,看不懂
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2019-12-23 17:21:34 | 显示全部楼层
请问为什么调用__new__方法必须要在括号中加入cls参数,而调用实例方法时候,不需要再括号中再加入self参数
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-11 16:51

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表