我正在使用基类构造函数作为工厂并在此构造函数/工厂中更改类以选择适当的类 - 这种方法是很好的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__
在子类中定义,该怎么办?