如何使用zope.interface.directlyProvides与类型的构建实例(字典,字符串,...)

时间:2014-07-17 10:42:09

标签: python zope.interface

我有一堆字典,我想用类型信息进行注释,以便以后能够为它们获取适配器。在下面的示例中,失败的情况是我想要做的,另一个情况显示工作版本。是否有可能在不引入额外对象的情况下使第一个版本正常工作?创建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)

2 个答案:

答案 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的子类是一种解决方法。

但是,你必须真正问一下,在这里调整词典是否是解决问题的最佳方法。