RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1589619
Accepted
Corkscrew
Corkscrew
Asked:2024-08-06 00:34:20 +0000 UTC2024-08-06 00:34:20 +0000 UTC 2024-08-06 00:34:20 +0000 UTC

为什么装饰器以不同的顺序执行?

  • 772

我正在和装饰者打交道。按照理论,它们是自下而上应用的,也就是说,最接近被装饰函数的描述的是第一个。然而,代码

'''case 1'''
import functools

def bold(func):
    @functools.wraps(func)
    def wrapper(*args, **kwargs):
        return '<b>' + func(*args, **kwargs) + '</b>'
    return wrapper

def italic(func):
    @functools.wraps(func)
    def wrapper(*args, **kwargs):
        return '<i>' + func(*args, **kwargs) + '</i>'
    return wrapper

strng = '1123123123'

@bold
@italic
def greet(string):
    return string

print(greet(strng))

    """Case 2:"""

    def dec1(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
                print(f'Первый декоратор')
                value = func(*args, **kwargs)
                return value
        return wrapper
    
    def dec2(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print(f'Второй декоратор')
            value = func(*args, **kwargs)
            return value
        return wrapper
    
    @dec1
    @dec2
    def function(string):
        return string
    
    print(function('fdsfsdf'))

给我结果

<b><i>Hello world!</i></b>
第一个装饰器
第二个装饰器
fdsfsdf

问题是为什么在第一种情况下装饰器像理论上一样使用,首先是斜体,然后是粗体,从下到上。在第二种情况下,首先是 dec1,它在顶部,然后是 dec2,它在下面?我不明白装饰器和函数的描述有何不同,不同行为的原因是什么?

python
  • 3 3 个回答
  • 57 Views

3 个回答

  • Voted
  1. AnnaBazueva - SPAM
    2024-08-06T01:08:08Z2024-08-06T01:08:08Z

    这是幻觉!

    案例3:

    import functools
    
    
    def dec1(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print(f'Первый декоратор')
            return 'Первый ' + func(*args, **kwargs) + ' Первый'
        return wrapper
    
    def dec2(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print(f'Второй декоратор')
            return 'Второй ' + func(*args, **kwargs) + ' Второй'
        return wrapper
    
    @dec1
    @dec2
    def function(string):
        return string
    
    print(function('DFFFFFFFFD'))
    

    结论:

    Первый декоратор
    Второй декоратор
    Первый Второй DFFFFFFFFD Второй Первый
    

    那些。:

    • 传递给的参数是自下而上function('DFFFFFFFFD')包装的,
    • 按从上到下的wrapper(*args, **kwargs)顺序运行的代码
    • 0
  2. CrazyElf
    2024-08-06T01:26:41Z2024-08-06T01:26:41Z

    只需添加更多打印功能,一切就会变得清晰:

    import functools
    
    def bold(func):
        print('Декоратор bold')
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print('Вызов bold')
            return '<b>' + func(*args, **kwargs) + '</b>'
        return wrapper
    
    def italic(func):
        print('Декоратор italic')
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print('Вызов italic')
            return '<i>' + func(*args, **kwargs) + '</i>'
        return wrapper
    
    strng = '1123123123'
    
    @bold
    @italic
    def greet(string):
        print('Вызов greet')
        return string
    
    print(greet(strng))
    

    结论:

    Декоратор italic
    Декоратор bold
    Вызов bold
    Вызов italic
    Вызов greet
    <b><i>1123123123</i></b>
    

    创建装饰器的代码实际上是以相反的顺序调用的。装饰者相互包裹。最底层是要装饰的函数,最后一个装饰器包装了这个函数,倒数第二个装饰器包装了最后一个装饰器,等等。

    但是,当整个嵌套娃娃(感谢斯坦尼斯拉夫·沃洛达尔斯基(Stanislav Volodarskiy)提出这个术语)被执行时,它已经从上到下执行了,因为外面有第一个装饰者,他是最后一个包装它的人。这些层以与缠绕方式相反的顺序显露出来。第一个装饰器调用第二个装饰器,依此类推,最后一个装饰器调用被装饰的函数。

    嗯,也就是说,这个想法正是装饰器应该按照直接顺序执行,即它们的编写方式。为此,创建装饰器的代码必须以相反的顺序执行,这就是 Python 中的做法。

    • 0
  3. Best Answer
    Stepan Roze
    2024-08-06T01:45:11Z2024-08-06T01:45:11Z

    您提出了一个关于 Python 中装饰器的使用的有趣观点,并且您对两个示例之间行为差异的观察确实需要一些解释。装饰器总是以自下而上的方式应用,这意味着最接近函数的装饰器首先被应用。

    让我们更详细地看看您的两个示例:

    示例1:

    import functools
    
    def bold(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            return '<b>' + func(*args, **kwargs) + '</b>'
        return wrapper
    
    def italic(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            return '<i>' + func(*args, **kwargs) + '</i>'
        return wrapper
    
    strng = '1123123123'
    
    @bold
    @italic
    def greet(string):
        return string
    
    print(greet(strng))
    

    在此示例中,装饰器的应用如下:

    1. 首先应用于italic函数greet,我们得到一个将结果包装greet在 中的函数<i>。
    2. 然后将其应用于bold上一步的结果,将结果包装在 中<b>。

    因此,结果是:'<b><i>1123123123</i></b>'。

    示例2:

    import functools
    
    def dec1(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print(f'Первый декоратор')
            value = func(*args, **kwargs)
            return value
        return wrapper
    
    def dec2(func):
        @functools.wraps(func)
        def wrapper(*args, **kwargs):
            print(f'Второй декоратор')
            value = func(*args, **kwargs)
            return value
        return wrapper
    
    @dec1
    @dec2
    def function(string):
        return string
    
    print(function('fdsfsdf'))
    

    在这个例子中,装饰器的顺序是相同的:

    1. 首先应用于dec2函数function,我们得到一个在调用之前打印“Second Decorator”的函数function。
    2. 然后将其应用于dec1上一步的结果,将调用包装在附加的“First Decorator”打印中。

    所以结果是:

    • 调用first dec1,打印“Firstdecorator”。
    • 然后在里面调用dec2打印“Second Decorator”。
    • 然后调用原来的函数function。

    结果:

    Первый декоратор
    Второй декоратор
    fdsfsdf
    

    结论

    这两个示例都遵循相同的装饰器使用规则:自下而上。之所以出现感知上的差异,是因为在第一个示例中,装饰器修改了函数的返回值,而在第二个示例中,它们在调用函数之前添加了副作用(打印到控制台)。因此,在这两种情况下,装饰器都可以正常工作并以相同的顺序应用,它们只是对最终结果产生不同的影响。

    • 0

相关问题

  • 是否可以以某种方式自定义 QTabWidget?

  • telebot.anihelper.ApiException 错误

  • Python。检查一个数字是否是 3 的幂。输出 无

  • 解析多个响应

  • 交换两个数组的元素,以便它们的新内容也反转

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5