从超类实例化子类

时间:2013-01-06 19:53:10

标签: python

我需要从超类中实例化一个子类。

class Superclass(object):
    @staticmethod
    def get_instance():
        #This should return an instance of subclass1 or subclass2

class Subclass1(Superclass):
    pass
class Subclass2(Superclass):
    pass

我想写:

Subclass1.get_instance()

获取Subclass1的实例,并且Subclass2

的实例相同

3 个答案:

答案 0 :(得分:9)

使用@classmethod代替@staticmethod

class Superclass(object):
    @classmethod
    def get_instance(cls):
        #This should return an instance of subclass1 or subclass2
        return cls()

class Subclass1(Superclass):
    pass
class Subclass2(Superclass):
    pass

答案 1 :(得分:0)

如果要访问给定类的子类列表,可以使用__subclasses__方法:

>>> class MyClass(object):
...     pass
... 
>>> class Subclass1(MyClass):
...     pass
... 
>>> class Subclass2(MyClass):
...     pass
... 
>>> MyClass.__subclasses__()
[<class '__main__.Subclass1'>, <class '__main__.Subclass2'>]

如果您已经知道现有的子类,则可以直接实例化它们:

>>> class MyClass(object):
...     def getInstance(self):
...             return Subclass1()
... 
>>> class Subclass1(MyClass): pass
... 
>>> MyClass().getInstance()
<__main__.Subclass1 object at 0x1e72d10>

无论如何,我猜测您正在尝试实现Singleton模式,在这种情况下,我认为您根本不应该使用getInstance方法。只需使用以下方法实现它:

  • 一个模块,因为模块是单例
  • 如果实例已存在,则重新实现__new__以返回旧实例
  • 使用某些元类

还有很多方法可以做到这一点。

如果这不是你的目标,那么也许你应该改变设计,因为超类通常不必了解子类。

答案 2 :(得分:0)

如果您不能在类方法中调用selfclass(),那么正确的方法是定义(或假设)工厂方法,并让您的超类方法在self上调用该工厂方法:

class Mixinator(object):

      @classmethod # this is not really necessary, but it means you don't need an instance to start with
      def factoryuser(selfclass, *args):
          return selfclass.factory(*args)


      #only define factory if you want Mixinator to be usable on its own
      @classmethod
      def factory(selfclass, *args):
          return selfclass(*args)  # this is a stub.

显然,儿童班应该需要在他们的factory中更多地参与其中,以使其有价值。我不确定是否有任何用例通过将factory替换为适当的__new__来解决,但这更明确。