我有一堆字典,我想用类型信息进行注释,以便以后能够为它们获取适配器。在下面的示例中,失败的情况是我想要做的,另一个情况显示工作版本。是否有可能在不引入额外对象的情况下使第一个版本正常工作?创建dicts的代码不容易改变,因此我正在寻找添加某些类型信息的最简单且非侵入性的方式。
from zope.interface import Interface, implements, directlyProvides
from zope.interface.registry import Components
registry = Components()
class IA(Interface):
pass
# this one fails
data = {}
directlyProvides(data, IA)
# this way it works
class X(dict):
pass
data = X()
directlyProvides(data, IA)
答案 0 :(得分:1)
zope.interface
依赖于在各种对象和类上设置一些魔术属性的能力。这就是发生的事情
>>> class Dict(dict):
... pass
...
>>> getattr(Dict, '__provides__', None)
无。现在尝试做你做的事。
>>> from zope.interface import Interface, implements, directlyProvides
>>> class IA(Interface):
... pass
...
>>> data = Dict()
>>> directlyProvides(data, IA)
尝试再次抓住__provides__
>>> getattr(Dict, '__provides__', None)
<zope.interface.declarations.ClassProvides object at 0x7f133cab48d0>
这对内置类型失败的原因是因为你无法在其中任何一个上设置属性(通过标准方式)。
>>> setattr(dict, '__provides__', None)
Traceback (most recent call last):
File "<console>", line 1, in <module>
TypeError: can't set attributes of built-in/extension type 'dict'
因此,如果要在某些标准类型的实例上分配一些标记接口,它将失败。只需按原样对它们进行子类化,或创建真实对象,这更像是Zope组件体系结构的设计用途。
答案 1 :(得分:1)
您不能使用接口信息注释Python内置类型;你根本无法添加所需的属性。
您可以注册该类型的适配器(因此没有实现接口):
>>> from zope.interface.registry import Components
>>> from zope.interface import Interface
>>> registry = Components()
>>> class IA(Interface):
... pass
...
>>> data = {}
>>> registry.registerAdapter(lambda o: 'adapter from dict to IA', [dict], IA)
>>> registry.queryAdapter(data, IA)
'adapter from dict to IA'
不幸的是,您不能为实例执行此操作:
>>> registry.registerAdapter(lambda o: 'adapter from data to IA', [data], IA)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/Users/mj/Development/venvs/stackoverflow-2.7/lib/python2.7/site-packages/zope/interface/registry.py", line 186, in registerAdapter
required = _getAdapterRequired(factory, required)
File "/Users/mj/Development/venvs/stackoverflow-2.7/lib/python2.7/site-packages/zope/interface/registry.py", line 432, in _getAdapterRequired
raise TypeError("Required specification must be a "
TypeError: Required specification must be a specification or class.
这意味着如果你真的必须调整特定的词典,那么你的选择是有限的。使用dict
的子类是一种解决方法。
但是,你必须真正问一下,在这里调整词典是否是解决问题的最佳方法。