heidern0612 发表于 2018-12-28 15:28:20

【第041讲心得】【认识__new__构造函数】

本帖最后由 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__ 方法是咩?(原谅我调皮放荡不羁爱自由{:10_297:} )

当你实例化一个对象的时候,首先就会执行__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__。





arley520 发表于 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)

四点好 发表于 2019-4-6 11:17:35

__new__用来做单例用的,楼主说明的很不错。

过默 发表于 2019-4-7 10:42:59

什么,看不懂

jiajiaself 发表于 2019-12-23 17:21:34

请问为什么调用__new__方法必须要在括号中加入cls参数,而调用实例方法时候,不需要再括号中再加入self参数
页: [1]
查看完整版本: 【第041讲心得】【认识__new__构造函数】