我正在使用基类构造函数作为工厂并在此构造函数/工厂中更改类以选择适当的类 - 这种方法是很好的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
有了这段代码,我可以通过基类Project创建ProjectLocal/ProjectRemote的实例:
project = Project('http://example.com') project.do_something()
我知道另一种方法是使用将返回基于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()
我的第一种方法是品味问题还是有一些隐藏的陷阱?
你不应该为此需要元类.看看这个__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
我会坚持工厂功能的方法.它是非常标准的python,易于阅读和理解.您可以通过多种方式使其更通用,以便处理更多选项,例如将鉴别器函数和结果映射传递给类.
如果第一个例子起作用,那就更多的是运气而不是设计.如果您想__init__
在子类中定义,该怎么办?