python - 为什么 *args 和 **kwargs 似乎在类装饰器中消失了?

标签 python python-3.x decorator

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

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()
  • 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})

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

关于python - 为什么 *args 和 **kwargs 似乎在类装饰器中消失了?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48627036/

相关文章:

python - Python 中的元组是不可变的吗?

python - 如何基于两个或多个其他变量创建 Pandas 数据框变量/列?

python - 在 python 中退出主循环

java - IO 的 GoF 装饰器模式用例和示例

python - 您如何将Elasticsearch Ingest附件处理器插件与Python软件包elasticsearch-dsl结合使用

python-3.x - ImportError : cannot import name from __init__. py

python - 如何使用 praw 从 reddit 下载视频

python-3.x - Python - 基于不同列转置/透视一列

python - 如何编写适当的装饰器来改变类

angular - ionic 2 应用程序中的装饰器@page、@component 和@app 是什么意思?