为什么我的想法在python2中不起作用?

wim*_*wim 7 python decorator python-2.x python-3.x python-decorators

这是一个可以改变键的dict子类的想法.这是一个简单的自包含示例,它就像一个dict但不区分大小写的str键.

from functools import wraps

def key_fix_decorator(f):
    @wraps(f)
    def wrapped(self, *args, **kwargs):
        if args and isinstance(args[0], str):
            args = (args[0].lower(),) + args[1:]
        return f(self, *args, **kwargs)
    return wrapped

class LowerDict(dict):
    pass

for method_name in '__setitem__', '__getitem__', '__delitem__', '__contains__', 'get', 'pop', 'setdefault':
    new_method = key_fix_decorator(getattr(LowerDict, method_name))
    setattr(LowerDict, method_name, new_method)
Run Code Online (Sandbox Code Playgroud)

dev注意:如果你复制我的代码用于你自己的用途,你应该实现LowerDict.__init__检查任何关键的冲突 - 我没有打算包括为了这个问题的目的

在python3上它似乎一切正常:

>>> d = LowerDict(potato=123, spam='eggs')
>>> d['poTATo']
123
>>> d.pop('SPAm')
'eggs'
>>> d['A']
# KeyError: 'a'
Run Code Online (Sandbox Code Playgroud)

在python2中它甚至不导入,这里是回溯:

  File "/tmp/thing.py", line 15, in <module>
    new_method = key_fix_decorator(getattr(LowerDict, method_name))
  File "/tmp/thing.py", line 4, in key_fix_decorator
    @wraps(f)
  File "/usr/lib/python2.7/functools.py", line 33, in update_wrapper
    setattr(wrapper, attr, getattr(wrapped, attr))
AttributeError: 'wrapper_descriptor' object has no attribute '__module__'
Run Code Online (Sandbox Code Playgroud)

可能是什么问题呢?除了str/ basestringthing 之外,我看不到任何特定于版本的代码,这只是一个小细节,而不是代码破解问题.

Mar*_*ers 4

Python 3 中的版本functools.wraps()可以处理函数对象,其复制的一些属性缺失;Python 2 中的则不能。这是因为问题 #3445仅针对 Python 3 进行了修复;的方法dict是在C代码中定义的并且没有__module__属性。

省略@wraps(f)装饰器也能让一切在 Python 2 中正常工作:

>>> def key_fix_decorator(f):
...     def wrapped(self, *args, **kwargs):
...         if args and isinstance(args[0], str):
...             args = (args[0].lower(),) + args[1:]
...         return f(self, *args, **kwargs)
...     return wrapped
... 
>>> class LowerDict(dict):
...     pass
... 
>>> for method_name in '__setitem__', '__getitem__', '__delitem__', '__contains__', 'get', 'pop', 'setdefault':
...     new_method = key_fix_decorator(getattr(LowerDict, method_name))
...     setattr(LowerDict, method_name, new_method)
... 
>>> d = LowerDict(potato=123, spam='eggs')
>>> d['poTATo']
123
>>> d.pop('SPAm')
'eggs'
>>> d['A']
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 5, in wrapped
KeyError: 'a'
Run Code Online (Sandbox Code Playgroud)

您可以复制足够多wraps手动操作:

def key_fix_decorator(f):
    def wrapped(self, *args, **kwargs):
        if args and isinstance(args[0], str):
            args = (args[0].lower(),) + args[1:]
        return f(self, *args, **kwargs)
    wrapped.__name__ = f.__name__
    wrapped.__doc__ = f.__doc__
    return wrapped
Run Code Online (Sandbox Code Playgroud)

wraps或限制尝试复制的属性:

def key_fix_decorator(f):
    @wraps(f, assigned=('__name__', '__doc__'))
    def wrapped(self, *args, **kwargs):
        if args and isinstance(args[0], str):
            args = (args[0].lower(),) + args[1:]
        return f(self, *args, **kwargs)
    return wrapped
Run Code Online (Sandbox Code Playgroud)

您实际上不需要__module__在这里更新属性;这主要只对内省有用。