【问题标题】:Why do *args and **kwargs seem to disappear in class decorator?为什么 *args 和 **kwargs 似乎在类装饰器中消失了?
【发布时间】:2018-02-05 16:26:35
【问题描述】:

用装饰器练习,发现这种行为很奇怪:

def test_decorator(cls, *args, **kwargs):
    print (args, kwargs)
    def build(*args, **kwargs):
        print (args, kwargs)
        return cls(*args, **kwargs)
    return build

@test_decorator
class Test:
    def __init__(self, *args, **kwargs):
        self.args = args
        self.kwargs = kwargs

t = Test(1, 2, 3, val = 4)
print (t.args, t.kwargs)

# output
# () { }
# (1, 2, 3) {'val' = 4}
# (1, 2, 3) {'val' = 4}

为什么装饰器中的第一个print 显示的是空容器?此外,如果我将build() 定义为:

def build():
    return cls(*args, **kwargs)

我知道它会因为nested function scoping 而失败。我只是不确定为什么在调用build 之前它们不存在并突然返回范围。

【问题讨论】:

  • 它们是要构建的参数,传递给test_decorator 的唯一内容是cls。不清楚你认为什么消失了。

标签: python python-3.x decorator


【解决方案1】:

这里有两个不同的可调用对象:

  • test_decorator()
  • test_decorator()build() 返回的包装器。

您混淆了两者。

第一个是用类调用的,因为

@test_decorator
class Test:
    # ...

真的只是

class Test:
    # ...
Test = test_decorator(Test)

该调用仅传递一个参数,即被修饰的类,该类被分配给cls 名称。该调用的 argskwargs 参数保持为空。

当您调用Test(...) 时,您实际上是在调用build(...)。该调用传递了由本地argskwargs 对象捕获的参数,并传递给cls(...)(它引用了原始类对象)。这些参数并没有丢失,它们显然被传递给__init__ 方法,并且正确设置了同名的实例属性。

要区分不同的包罗万象的参数,请先给它们起不同的名称并增加您的 print() 输出:

def test_decorator(cls, *decorator_args, **decorator_kwargs):
    print('Decorator called with ({!r}, *{!r}, **{!r})'.format(
        cls, decorator_args, decorator_kwargs))
    def build(*build_args, **build_kwargs):
        print('build() wrapper called with (*{!r}, **{!r})'.format(
            build_args, build_kwargs))
        print('The decorator was originally called with ({!r}, *{!r}, **{!r})'.format(
            cls, decorator_args, decorator_kwargs))
        return cls(*build_args, **build_kwargs)
    return build

现在输出变成:

>>> @test_decorator
... class Test:
...     def __init__(self, *args, **kwargs):
...         self.args = args
...         self.kwargs = kwargs
...
Decorator called with (<class '__main__.Test'>, *(), **{})
>>> t = Test(1, 2, 3, val = 4)
build() wrapper called with (*(1, 2, 3), **{'val': 4})
The decorator was originally called with (<class '__main__.Test'>, *(), **{})
>>> t.args, t.kwargs
((1, 2, 3), {'val': 4})

请注意,当class 语句执行时,会产生Decorator call with ... 输出,而t = Test(...) 调用触发了名为...的build() 包装器。 输出。

【讨论】:

  • 那么传递给装饰器的类是否保留其参数?
  • @pstatix:是的,确实如此。您已经在自己的输出中看到了这一点。
  • 非常好的文章;从这里我想约定是每个装饰器都有一个单一的职责,这意味着它应该只有一个包装函数内部?
  • @pstatix:不一定。装饰器是一个非常简单的契约:调用函数或类上方的@&lt;something&gt; 语法并传入函数或类。装饰器返回的任何都分配给原始名称。除了合同之外,您还有很大的灵活性。
  • @pstatix:装饰器可以只返回原始对象。这就是有多少 Web 框架(如 Django 和 Flask)使用装饰器来注册函数来处理请求:它们使用信息更新注册表并返回原始函数,否则保持不变。
猜你喜欢
  • 2018-08-28
  • 2013-09-25
  • 2016-09-02
  • 1970-01-01
  • 2016-04-20
  • 1970-01-01
  • 1970-01-01
  • 2014-04-26
  • 2021-07-26
相关资源
最近更新 更多