一文彻底读懂Python装饰器

简介: 一文彻底读懂Python装饰器

一文彻底读懂Python装饰器


装饰器主要用途是:

不修改函数源码的前提下,添加额外的功能。

如果你有Java开发经验,你会发现,Python中的装饰器其实就类似于Java的注解。好的,废话不多说,进入正题。

我们假想如下一个场景:

测试一个现有的函数的执行耗时,要求是不能修改原始函数块内代码。

1 探索无装饰器的场景实现

1.1 简单初版实现

对于如上需求,我们很轻松会想到一个方案:将函数及其相关参数对象传入一个新定义的函数,在新定义的函数做耗时测试,具体如下:

import time
def target_func(a, b, c):
    time.sleep(2)
    return a**b + c
def test_time(func, a, b, c):
    t = time.time()
    out = func(a, b, c)
    print(time.time() - t)
  return out
test_time(target_func, 99, 199, 9)

1.2 升级版实现

上面这种方案还存在一些缺点:参数不灵活,只能测试有3个参数的函数。因此,升级版如下:

def test_time(func, *args, **kwargs):
    t = time.time()
    out = func(*args, **kwargs)
    print(time.time() - t)
  return out
test_time(target_func, 99, 199, 9)

1.3 进阶版实现

上面的封装还是存在缺点,即破坏了用户的开发思路。用户设计代码时,已经将target_func作为某项功能实现对待。如果转而去执行test_time函数,那么需要传入target_func函数对象作为test_time的参数。在某种程度上说,已经破坏了代码。因此,最好让使用者无感植入代码。具体实现如下:

import time
def test_time(func):
    def wrapper(*args, **kwargs):
        t = time.time()
        out = func(*args, **kwargs)
        print(time.time() - t)
        return out
    return wrapper
# 提前封装好函数
my_func = test_time(target_func)
my_func(99, 199, 9)

倒数第二行对函数做了装饰,经过装饰后的函数可以直接调用,并且调用者是无感的。

2 使用装饰器

2.1 无参装饰器

2.1.1 直接使用@符号

其实1.3小节的进阶版实现就已经是一个装饰器了,读者可能注意到,虽然说1.3中的最后一行代码是无感调用的,但是其实倒数第二行手动去对目标函数做了封装。这一行代码无疑非常影响代码的极简风格,因此,在Python中,这一行代码可以直接使用装饰器来取代:

import time
def test_time(func):
    def wrapper(*args, **kwargs):
        t = time.time()
        out = func(*args, **kwargs)
        print(time.time() - t)
        return out
    return wrapper
@test_time
def target_func(a, b, c):
    time.sleep(2)
    return a**b + c
target_func(99, 199, 9)

仔细看target_func函数,可以看到,在函数定义前加了装饰器@test_time,这一行等价于:

在执行target_func函数之前,Python解释器会先执行test_time函数,并将返回的函数对象在原来执行的位置通过“偷梁换柱”替换掉。

简而言之,2.1.1中的代码与1.3中的代码是等价的。

2.1.2 使用functools.wraps

2.1.1中的wrapper函数其实有个非常大的问题!即原始函数被偷梁换柱了。在一些业务或者框架中,如果底层需要对函数进行判断,那么将会引来一个BUG,我们做个测试:

import time
def test_time(func):
    def wrapper(*args, **kwargs):
        '''
        这里是wrapper函数的注释文档
        '''
        t = time.time()
        out = func(*args, **kwargs)
        print(time.time() - t)
        return out
    return wrapper
@test_time
def target_func(a, b, c):
    '''
    这里是target_func函数的注释文档
    '''
    time.sleep(2)
    return a**b + c
print(target_func.__name__, target_func.__doc__)

输出结果如下:

wrapper 
        这里是wrapper函数的注释文档

可以看到,明显当前target_func函数已经不是当年那个target_func函数。为了解决这个问题,python中引入了装饰器functools.wraps,只需在wrapper函数中加入functools.wraps装饰器即可:

import time
from functools import wraps
def test_time(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        '''
        这里是wrapper函数的注释文档
        '''
        t = time.time()
        out = func(*args, **kwargs)
        print(time.time() - t)
        return out
    return wrapper
@test_time
def target_func(a, b, c):
    '''
    这里是target_func函数的注释文档
    '''
    time.sleep(2)
    return a**b + c
print(target_func.__name__, target_func.__doc__)

输出结果如下:

target_func 
    这里是target_func函数的注释文档

2.2 有参数装饰器

2.1中使用装饰器时,没有提供额外的参数,有时在装饰器中不仅仅需要目标函数对象,也需要额外的其他参数。

import time
def test_time(msg):
    def test_time_inner(func):
      @wraps(func)
        def wrapper(*args, **kwargs):
            print("before func:", msg)
            t = time.time()
            out = func(*args, **kwargs)
            print(time.time() - t)
            return out
        return wrapper
    return test_time_inner
@test_time("测试信息")
def target_func(a, b, c):
    time.sleep(2)
    return a**b + c
target_func(99, 199, 9)

输出结果如下:

before func: 测试信息
2.009385585784912

可以看到,如果需要给装饰器传入参数,那么需要将装饰器封装为3层函数。其中:

最外层函数接收装饰器参数。

内部两层为普通的装饰器定义方式。

3 装饰器类

我们知道,在python中,类实例也是callable的,即类实例也可以像函数一样调用,且实际调用的是类实例的__call__函数。既然如此,类实例也可以完成函数能做的事情:

import time
from functools import wraps
class TestTime:
    def __init__(self, msg):
        self.msg = msg
    def __call__(self, func):
      @wraps(func)
        def wrapper(*args, **kwargs):
            print("before func:", self.msg)
            t = time.time()
            out = func(*args, **kwargs)
            print(time.time() - t)
            return out
        return wrapper
@TestTime("测试信息")
def target_func(a, b, c):
    time.sleep(2)
    return a**b + c
target_func(99, 199, 9)

输出结果如下:

before func: 测试信息
2.007122278213501


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