探索Python中的装饰器:从基础到高级

简介: 【10月更文挑战第11天】 在这篇文章中,我们将深入探讨Python装饰器的强大功能和灵活应用。装饰器是Python中一个非常有趣的特性,它允许我们修改或增强函数的行为,而无需直接修改函数本身的代码。通过使用装饰器,我们可以实现横切关注点(AOP)的编程范式,提高代码的可重用性和模块化。本文将介绍装饰器的基本概念、使用方法以及如何创建自定义装饰器。同时,我们还将探讨装饰器的一些高级用法,如带参数的装饰器、多层装饰器和偏函数装饰器等。

一、装饰器的基本概念和使用

  1. 定义装饰器
  2. 无参装饰器
  3. 有参装饰器
  4. 装饰器嵌套使用

二、装饰器的高级应用

  1. 带参数的装饰器
  2. 多层装饰器
  3. 偏函数作为装饰器
  4. 类装饰器与装饰器类
  5. 动态装饰器

三、总结与展望

  1. 总结
  2. 展望

正文:
一、装饰器的基本概念和使用

  1. 定义装饰器
    装饰器本质上是一个接受函数作为参数的高阶函数。它可以对传入的函数进行包装,并返回一个新的函数。这个新函数通常会包含原函数的调用,并在其基础上添加一些额外的功能。
  2. 无参装饰器
    最简单的装饰器形式是无参装饰器。这种装饰器不需要额外的参数,只需定义一个外部函数,该函数接受一个函数作为参数并返回一个新函数。例如:
    ```python
    def my_decorator(func):
    def wrapper():
     print("Something is happening before the function is called.")
     func()
     print("Something is happening after the function is called.")
    
    return wrapper

@my_decorator
def say_hello():
print("Hello!")

say_hello()

3. **有参装饰器**
   有参装饰器是指装饰器本身也接受参数。要实现这样的装饰器,可以使用嵌套函数的方式。例如:
```python
def repeat(num):
    def decorator(func):
        def wrapper(*args, **kwargs):
            for _ in range(num):
                func(*args, **kwargs)
        return wrapper
    return decorator

@repeat(3)
def say_hello(name):
    print(f"Hello, {name}!")

say_hello("Alice")
  1. 装饰器嵌套使用
    在一个函数上可以应用多个装饰器,这被称为装饰器的嵌套使用。当有多个装饰器作用于同一个函数时,它们的执行顺序是从内到外依次执行的。例如:
    ```python
    def bold(func):
    def wrapper(args, *kwargs):
     return f"<b>{func(*args, **kwargs)}</b>"
    
    return wrapper

def italic(func):
def wrapper(args, **kwargs):
return f"{func(
args, **kwargs)}"
return wrapper

@bold
@italic
def greet(name):
return f"Hello, {name}!"

print(greet("Alice"))


二、装饰器的高级应用
1. **带参数的装饰器**
   在某些情况下,我们希望为装饰器本身提供参数。要实现这一目标,可以先定义一个返回装饰器的函数。例如:
```python
def valid_age(age):
    def decorator(func):
        def wrapper(person):
            if person['age'] >= age:
                return func(person)
            else:
                return f"Sorry, you must be at least {age} years old."
        return wrapper
    return decorator

@valid_age(18)
def enter_movie(person):
    return f"Welcome to the movie, {person['name']}!"

print(enter_movie({"name": "Alice", "age": 20}))
print(enter_movie({"name": "Bob", "age": 17}))
  1. 多层装饰器
    多层装饰器指的是在一个函数上应用多个装饰器,这些装饰器之间可能存在嵌套关系。通过合理使用多层装饰器,可以实现更复杂的功能组合。例如:
    ```python
    def make_bold(fn):
    def wrapped():
     return f"<b>{fn()}</b>"
    
    return wrapped

def make_italic(fn):
def wrapped():
return f"{fn()}"
return wrapped

@make_bold
@make_italic
def say_hello():
return "Hello, World!"

print(say_hello())

3. **偏函数作为装饰器**
   偏函数是一种简化函数调用的方式。通过使用偏函数作为装饰器,可以避免重复编写相似的代码。例如:
```python
from functools import partial
from time import perf_counter as pc

def timing(f):
    def inner(*args, **kwargs):
        start = pc()
        result = f(*args, **kwargs)
        end = pc()
        print(f"Function {f.__name__} took {end - start:.6f} secs")
        return result
    return inner

@timing
def sum_list(lst):
    return sum(lst)

lst = [i for i in range(1, 1000000)]
sum_list(lst)
  1. 类装饰器与装饰器类
    除了函数式装饰器外,还可以使用类来实现装饰器功能。类装饰器具有更好的可扩展性和可维护性。例如:
    ```python
    class Timing:
    def init(self, func):

     self.func = func
    

    def call(self, args, *kwargs):

     start = pc()
     result = self.func(*args, **kwargs)
     end = pc()
     print(f"Function {self.func.__name__} took {end - start:.6f} secs")
     return result
    

@Timing
def sum_list(lst):
return sum(lst)

lst = [i for i in range(1, 1000000)]
sum_list(lst)

5. **动态装饰器**
   动态装饰器是指在运行时根据条件选择性地应用装饰器。这可以通过定义一个返回装饰器的函数来实现。例如:
```python
def maybe_decorator(condition):
    def decorator(func):
        def wrapper(*args, **kwargs):
            if condition:
                print("Decorator applied.")
            return func(*args, **kwargs)
        return wrapper
    return decorator

@maybe_decorator(True)
def greet(name):
    return f"Hello, {name}!"

print(greet("Alice"))
目录
相关文章
|
4天前
|
编解码 Java 程序员
写代码还有专业的编程显示器?
写代码已经十个年头了, 一直都是习惯直接用一台Mac电脑写代码 偶尔接一个显示器, 但是可能因为公司配的显示器不怎么样, 还要接转接头 搞得桌面杂乱无章,分辨率也低,感觉屏幕还是Mac自带的看着舒服
|
6天前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1551 8
|
1月前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
10天前
|
人工智能 Rust Java
10月更文挑战赛火热启动,坚持热爱坚持创作!
开发者社区10月更文挑战,寻找热爱技术内容创作的你,欢迎来创作!
658 25
|
6天前
|
存储 SQL 关系型数据库
彻底搞懂InnoDB的MVCC多版本并发控制
本文详细介绍了InnoDB存储引擎中的两种并发控制方法:MVCC(多版本并发控制)和LBCC(基于锁的并发控制)。MVCC通过记录版本信息和使用快照读取机制,实现了高并发下的读写操作,而LBCC则通过加锁机制控制并发访问。文章深入探讨了MVCC的工作原理,包括插入、删除、修改流程及查询过程中的快照读取机制。通过多个案例演示了不同隔离级别下MVCC的具体表现,并解释了事务ID的分配和管理方式。最后,对比了四种隔离级别的性能特点,帮助读者理解如何根据具体需求选择合适的隔离级别以优化数据库性能。
213 3
|
1天前
|
Python
【10月更文挑战第10天】「Mac上学Python 19」小学奥数篇5 - 圆和矩形的面积计算
本篇将通过 Python 和 Cangjie 双语解决简单的几何问题:计算圆的面积和矩形的面积。通过这道题,学生将掌握如何使用公式解决几何问题,并学会用编程实现数学公式。
102 59
|
13天前
|
Linux 虚拟化 开发者
一键将CentOs的yum源更换为国内阿里yum源
一键将CentOs的yum源更换为国内阿里yum源
673 5
|
2天前
|
Java 开发者
【编程进阶知识】《Java 文件复制魔法:FileReader/FileWriter 的奇妙之旅》
本文深入探讨了如何使用 Java 中的 FileReader 和 FileWriter 进行文件复制操作,包括按字符和字符数组复制。通过详细讲解、代码示例和流程图,帮助读者掌握这一重要技能,提升 Java 编程能力。适合初学者和进阶开发者阅读。
100 61
|
13天前
|
JSON 自然语言处理 数据管理
阿里云百炼产品月刊【2024年9月】
阿里云百炼产品月刊【2024年9月】,涵盖本月产品和功能发布、活动,应用实践等内容,帮助您快速了解阿里云百炼产品的最新动态。
阿里云百炼产品月刊【2024年9月】
|
2天前
vue3+Ts 二次封装ElementUI form表单
【10月更文挑战第8天】
107 57