在Python中使用基类构造函数作为工厂?

bia*_*lix 8 python factory

我正在使用基类构造函数作为工厂并在此构造函数/工厂中更改类以选择适当的类 - 这种方法是很好的python实践还是有更优雅的方法?

我试图阅读有关元类的帮助,但没有取得很大的成功.

这是我正在做的事情的例子.

class Project(object):
  "Base class and factory."
  def __init__(self, url):
      if is_url_local(url):
        self.__class__ = ProjectLocal
      else:
        self.__class__ = ProjectRemote
      self.url = url

class ProjectLocal(Project):
  def do_something(self):
    # do the stuff locally in the dir pointed by self.url

class ProjectRemote(Project):
  def do_something(self):
    # do the stuff communicating with remote server pointed by self.url
Run Code Online (Sandbox Code Playgroud)

有了这段代码,我可以通过基类Project创建ProjectLocal/ProjectRemote的实例:

project = Project('http://example.com')
project.do_something()
Run Code Online (Sandbox Code Playgroud)

我知道另一种方法是使用将返回基于url的类对象的fabric函数,然后代码看起来类似:

def project_factory(url):
      if is_url_local(url):
        return ProjectLocal(url)
      else:
        return ProjectRemote(url)

project = project_factory(url)
project.do_something()
Run Code Online (Sandbox Code Playgroud)

我的第一种方法是品味问题还是有一些隐藏的陷阱?

Bri*_*ian 18

你不应该为此需要元类.看看这个__new__方法.这将允许您控制对象的创建,而不仅仅是初始化,因此返回您选择的对象.

class Project(object):
  "Base class and factory."
  def __new__(cls, url):
    if is_url_local(url):
       return super(Project, cls).__new__(ProjectLocal, url) 
    else:
       return super(Project, cls).__new__(ProjectRemote, url) 

  def __init__(self, url):
    self.url = url
Run Code Online (Sandbox Code Playgroud)


小智 13

我会坚持工厂功能的方法.它是非常标准的python,易于阅读和理解.您可以通过多种方式使其更通用,以便处理更多选项,例如将鉴别器函数和结果映射传递给类.

如果第一个例子起作用,那就更多的是运气而不是设计.如果您想__init__在子类中定义,该怎么办?

  • +1:不要试图让基类做更多的事情而不是基类.工厂总是分开的. (5认同)