__new__构造函数的常用做法?

时间:2014-03-07 13:20:51

标签: python constructor

我知道(?)关于Python中__new__构造函数背后的理论,但我要问的是常见的做法 - 为什么这个构造函数真的(!)使用?< / p>

我已经阅读过关于初始化不可变对象(逻辑从__init__移到__new__),其他什么?工厂模式?

请再次注意区别:

  • 可以使用__new__任务 - 我不感兴趣
  • 使用__new__ 的任务 - 我是: - )

我不用Python写任何东西,我的知识来自阅读,而不是来自经验。

您可以在哪里实际回答问题:Common practice of new constructor?

1 个答案:

答案 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