我知道(?)关于Python中__new__
构造函数背后的理论,但我要问的是常见的做法 - 为什么这个构造函数真的(!)使用?< / p>
我已经阅读过关于初始化不可变对象(逻辑从__init__
移到__new__
),其他什么?工厂模式?
请再次注意区别:
__new__
任务 - 我不感兴趣__new__
的任务 - 我是: - )我不用Python写任何东西,我的知识来自阅读,而不是来自经验。
您可以在哪里实际回答问题:Common practice of new constructor?
答案 0 :(得分:3)
__new__
的要点是创建一个__init__
然后初始化的空对象实例。重新实现__new__
您可以完全控制您创建的实例,但实际上没有使用__init__
方法进行任何进一步处理。我可以给你两个有用的案例:使用智能构造函数从类的磁盘自动创建方法和反序列化。这些不是解决这两个问题的唯一方法。元类是另一种更灵活的方式,但是作为任何工具,您可能希望得到不同程度的复杂性。
自动创建方法
假设您想要一个具有给定属性集的类。您可以使用此代码
来控制如何初始化这些属性class Foo(object):
properties = []
def __new__(cls, *args):
instance = object.__new__(cls, *args)
for p in cls.properties:
setattr(instance, p, 0)
return instance
class MyFoo(Foo):
properties = ['bar', 'baz']
def __init__(self):
pass
f=MyFoo()
print dir(f)
您想要的属性直接初始化为零。您可以执行许多智能技巧,例如动态执行属性列表。实例化的所有对象都将具有这些方法。 Django Models中存在这种模式的更复杂的情况,您可以在其中声明字段并免费获得大量自动内容,这要归功于__new__
大哥,元类。
从磁盘反序列化
假设您有一个具有给定构造函数的类,该构造函数从对象填充类的字段,例如进程:
class ProcessWrapper(object):
def __init__(self, process):
self._process_pid = process.pid()
def processPid(self):
return self._process_pid
如果您现在将此信息序列化到磁盘并想要恢复它,则无法通过构造函数进行初始化。所以你编写了这样的反序列化函数,有效地绕过了你无法运行的__init__
方法。
def deserializeProcessWrapperFromFile(filename):
# Get process pid from file
process = ProcessWrapper.__new__()
process._process_pid = process_pid
return process