Python 装饰器“高级”使用

简介: 本文聚焦两个有意思的点1. 无参和有参装饰器。 @deco vs @deco(arg1,arg2)。2. 多层装饰器场景。

本文聚焦两个有意思的点

  1. 无参和有参装饰器。 @deco vs @deco(arg1,arg2)。
  2. 多层装饰器场景。

无参和有参装饰器

大部分文章,都会学习到无参有参装饰器写法。这里不赘述,直接上兼容括号和无括号注解 (无参 vs 有参),高级+灵活。

# 兼容有括号和无括号装饰器
def log3(func=None, /, *, text=None):
    """
    @log 装饰器(兼容有参和无参)
    三层嵌套.
    === f = log('text')(f) , f() 实际是 wrapper()
    :param text:
    :return:
    """
    def decorator(f):  # =~ 上面的 log 方法
        @functools.wraps(f)
        def wrapper(*args, **kw):
            t0 = time.time()
            print('[%s] start call %s at %s' % (text, f.__name__,
                                                time.strftime('%Y-%m-%d %H:%M:%S', time.localtime(t0))))
            # 调用目标方法本身
            ret = f(*args, **kw)
            t1 = time.time()
            print('[%s] end call %s at %s, cost time: %s s' % (text, f.__name__,
                                                               time.strftime(
                                                                   '%Y-%m-%d %H:%M:%S', time.localtime(t1)),
                                                               t1-t0))
            return ret
        return wrapper
    
    # 判断 func 是否是函数
    if inspect.isfunction(func):
        # 无括号
        print('deco nopars')
        return decorator(func)  # 记忆诀窍, 无括号, 要补上
    return decorator
@log3
def tarFun3_nopar():
    print("tar Fun3 executing - nopar")
@log3(text='log333333')
def tarFun3_haspar():
    print("tar Fun3 executing - haspar")
# tarFun3_nopar()
tarFun3_haspar()

要点:

装饰器代码里三层函数。里面判断是否是无参调用还是调用,准确的说是不带括号调用还是带括号调用,根据不同是否方式,返回对应的函数。


分隔符 /,* 含义:

  • / , 分割符号,前面表示只能是位置参数。这就限制了 func 参数只能位置参数传入,而不能通过 func=x 传入。
  • , 后面的参数只能是k-v传参。


有这两个分隔符参数限制,可以避免调用传参导致的错误。

多层装饰器场景

看例子

def deco1(func):
    print("deco1")
    def deco1_wrapper(*args, **kwargs):
        print("deco1 wrapper", func)
        return func(*args, **kwargs)
    return deco1_wrapper
def deco2(func):
    print("deco2")
    def deco2_wrapper(*args, **kwargs):
        print("deco2 wrapper", func)
        return func(*args, **kwargs)
    return deco2_wrapper
@deco1
@deco2
def mult_deco():
    print("mult_deco")
mult_deco()
# ---
deco2  -- 解释执行期间打印
deco1  -- 解释执行期间打印
deco1 wrapper <function deco2.<locals>.deco2_wrapper at 0x000002A514699C60>
deco2 wrapper <function mult_deco at 0x000002A514699BC0>
mult_deco

解释期间,注解由下到上解释(执行)。距离目标方法最近的注解先被解释执行,可以理解为由内而外。


要点:类似洋葱

  1. 解释期间,由内而外
  2. 执行期间,由外而内

剥洋葱 @functools.wrap

另外,例子中发现 deco1 装饰器里 func 打印字符是 deco2_wrapper 字样。很容易理解,根据洋葱定律,外层包裹的是内层的,故 deco1 包裹的自然是 deco2 deco2_wrapper 方法。


那该怎么保留或获取原始被包裹方法的信息呢?


来,让我们剥洋葱!


这就需要借助工具:@functools.wrap

import functools
def deco1(func):
    print("deco1")
    @functools.wraps(func)  # 如果我最外层使用, 可以不剥, 剥是为了将原始函数信息暴漏给更外层的
    def deco1_wrapper(*args, **kwargs):
        print("deco1 wrapper", func)  # 这里 func 由于里层装饰器剥了一遍洋葱, 故而这里拿到的是原始函数 mult_deco 信息
        return func(*args, **kwargs)
    return deco1_wrapper
def deco2(func):
    print("deco2")
    @functools.wraps(func)  # 虽然返回的deco2_wrapper, 但函数信息却是剥过洋葱后的, 即原始函数 mult_deco
    def deco2_wrapper(*args, **kwargs):
        print("deco2 wrapper", func)
        return func(*args, **kwargs)
    return deco2_wrapper
@deco1
@deco2
def mult_deco():
    print("mult_deco")
mult_deco()
deco2
deco1
deco1 wrapper <function mult_deco at 0x0000023387C19C60>
deco2 wrapper <function mult_deco at 0x0000023387C19BC0>
mult_deco


@functools.wrap 详细原理,这里不深究了,只附带介绍下它的现象。


要点:自定义装饰器推荐必带 @functools.wrap 装饰器。


作者:Nisus

链接:https://juejin.cn/post/7398520872179843072

相关文章
|
8天前
|
Python
深入理解Python装饰器:从入门到实践####
本文旨在通过简明扼要的方式,为读者揭开Python装饰器的神秘面纱,从基本概念、工作原理到实际应用场景进行全面解析。不同于常规的摘要仅概述内容概要,本文将直接以一段精炼代码示例开篇,展示装饰器如何优雅地增强函数功能,激发读者探索兴趣,随后深入探讨其背后的机制与高级用法。 ####
38 11
|
5天前
|
设计模式 缓存 开发者
深入浅出Python装饰器
【10月更文挑战第39天】本文将通过浅显易懂的语言和生动的比喻,带你探索Python中一个神奇而又强大的特性——装饰器。我们将一起揭开装饰器的神秘面纱,了解它的工作原理,并通过实际代码示例学习如何应用它来美化我们的代码。无论你是编程新手还是有经验的开发者,这篇文章都将为你打开一扇新的大门,让你的代码更加优雅和高效。
|
5天前
|
缓存 测试技术 数据库
深入理解Python中的装饰器
在本文中,我们将探讨Python语言中一个强大而灵活的特性——装饰器。装饰器允许开发者在不修改原有函数或方法代码的情况下增加额外的功能,这大大提高了代码的复用性和可读性。通过具体示例和应用场景的讲解,本篇文章旨在为读者提供一个关于如何使用装饰器的全面指南,包括装饰器的定义、使用场景、以及如何自定义装饰器等内容。
|
10天前
|
设计模式 Python
掌握Python中的装饰器
【10月更文挑战第34天】装饰器是Python中一种强大的工具,它允许我们在不修改原函数代码的情况下增加其功能。本文通过简单易懂的语言和实例,引导你理解装饰器的概念、种类及其应用,帮助你在编程实践中灵活使用这一高级特性。
|
6天前
|
缓存 监控 测试技术
Python中的装饰器:功能扩展与代码复用的利器###
本文深入探讨了Python中装饰器的概念、实现机制及其在实际开发中的应用价值。通过生动的实例和详尽的解释,文章展示了装饰器如何增强函数功能、提升代码可读性和维护性,并鼓励读者在项目中灵活运用这一强大的语言特性。 ###
|
9天前
|
缓存 开发者 Python
探索Python中的装饰器:简化代码,增强功能
【10月更文挑战第35天】装饰器在Python中是一种强大的工具,它允许开发者在不修改原有函数代码的情况下增加额外的功能。本文旨在通过简明的语言和实际的编码示例,带领读者理解装饰器的概念、用法及其在实际编程场景中的应用,从而提升代码的可读性和复用性。
|
5天前
|
设计模式 缓存 开发框架
Python中的装饰器:从入门到实践####
本文深入探讨了Python中装饰器的工作原理与应用,通过具体案例展示了如何利用装饰器增强函数功能、提高代码复用性和可读性。读者将学习到装饰器的基本概念、实现方法及其在实际项目开发中的实用技巧。 ####
18 3
|
5天前
|
Python
探索Python中的装饰器:简化代码,提升效率
【10月更文挑战第39天】在编程的世界中,我们总是在寻找使代码更简洁、更高效的方法。Python的装饰器提供了一种强大的工具,能够让我们做到这一点。本文将深入探讨装饰器的基本概念,展示如何通过它们来增强函数的功能,同时保持代码的整洁性。我们将从基础开始,逐步深入到装饰器的高级用法,让你了解如何利用这一特性来优化你的Python代码。准备好让你的代码变得更加优雅和强大了吗?让我们开始吧!
13 1
|
10天前
|
设计模式 缓存 监控
Python中的装饰器:代码的魔法增强剂
在Python编程中,装饰器是一种强大而灵活的工具,它允许程序员在不修改函数或方法源代码的情况下增加额外的功能。本文将探讨装饰器的定义、工作原理以及如何通过自定义和标准库中的装饰器来优化代码结构和提高开发效率。通过实例演示,我们将深入了解装饰器的应用,包括日志记录、性能测量、事务处理等常见场景。此外,我们还将讨论装饰器的高级用法,如带参数的装饰器和类装饰器,为读者提供全面的装饰器使用指南。
|
6天前
|
存储 缓存 监控
掌握Python装饰器:提升代码复用性与可读性的利器
在本文中,我们将深入探讨Python装饰器的概念、工作原理以及如何有效地应用它们来增强代码的可读性和复用性。不同于传统的函数调用,装饰器提供了一种优雅的方式来修改或扩展函数的行为,而无需直接修改原始函数代码。通过实际示例和应用场景分析,本文旨在帮助读者理解装饰器的实用性,并鼓励在日常编程实践中灵活运用这一强大特性。