Python class的__new__与__init__方法

通常我们按照Java或C++里的习惯, 称__init__方法为构造器方法, 其实这种说法并不准确, 因为类的实例在调用__init__方法之前已经在__new__方法里创建了. 所以说, 真正的构造器方法应该是__new__, 而__init__只是完成构造以外的初始化操作. 用Fluent Python里的伪代码描述实例的构造过程:
这里写图片描述
具体描述为: 先调用the_class.__new__(some_arg), 如果它的返回值new_objectthe_class类型的实例, 则继续调用the_class.__init__(new_object, some_arg), 否则直接返回.
值得注意的是__new__(cls, ...)是一个classmethod, 但它的定义并不需要用@classmethod, 因为它会被编译器特殊照顾.

大部分情况下我们只需要实现__init__方法而不需要实现__new__方法. 我暂时能想到的实现__new__方法的唯一用途是实现一个类似于Python内建类property的类, 它即是类, 也是一个修饰器:

class Foo():
    @property
    def name(self):
        return 'Foo'
    def get_age(self):
        return 10
    age = property(get_age)

print(Foo.name, Foo.age)
foo = Foo()
print(foo.name, foo.age)

输出:

<property object at 0x7fd135e604f8> <property object at 0x7fd135ef51d8>
Foo 10

借助__new__方法, 我们也可以实现类似的效果, 让一个类既当类又当修饰器, 当然, 这不是唯一的方法:

class cls_decorator(object):
    def __init__(self):
        print('__init__ is called')

    def __new__(cls, fn = None):
        if fn is not None and callable(fn):
            def new_fn(*args, **kwargs):
                print('decorated', fn.__name__, 'is called.')
                result = fn(*args, **kwargs)
                return result
            return new_fn
        else:
            return super().__new__(cls)
@cls_decorator
def fn(a, b, c):
    return 3

print('As a decorator:', fn(1, 2, c = 3))
print('As a class:', cls_decorator())

输出:

decorated fn is called.
As a decorator: 3
__init__ is called
As a class: <__main__.cls_decorator object at 0x7fd135e7b828>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值