Python装饰器使用方法全面梳理

作者:专注算法的马里奥学长 时间:2023-04-27 08:22:15 

1 装饰器背景知识

1.1 基本概念

装饰器(Decorator)是 Python 中一种函数或类,用来修饰其他函数或类。装饰器可以改变被装饰函数的行为,或者在调用被装饰函数之前和之后增加额外的操作。装饰器的语法是使用 @ 语法符,在函数定义之前增加装饰器函数的名称。

@decorator_func
def my_func():
   pass

1.2 应用场景

  • 代码重用:装饰器可以让我们在不更改原函数代码的情况下,为其添加额外的功能。

  • 日志记录:装饰器可以记录函数的调用日志,帮助我们追踪程序的运行情况。

  • 权限控制:装饰器可以用来实现函数级别的权限控制,只允许特定的用户访问特定的函数。

  • 缓存:装饰器可以用来缓存函数的返回值,避免重复计算。

  • 类型检查:装饰器可以用来在函数调用前检查参数的类型是否符合要求。

  • 装饰器可以让你在函数或类的定义中添加额外的逻辑,而不更改它们的实现。

2 简单的装饰器代码

def decorator_func(func):
   def wrapper():
       print("Before calling the function")
       func()
       print("After calling the function")
   return wrapper
@decorator_func
def my_func():
   print("Inside the function")
my_func()
# Output: Before calling the function
#         Inside the function
#         After calling the function

上面展示了最简单的装饰器示例代码。在代码中,我们建立了一个名为decorator_func的装饰器和一个名为my_func函数。

  • 装饰器外部的return必须为装饰器的内部函数,不含括号。通过代码结构可以看出,装饰器本身也是一个闭包。

  • 在定义装饰器decorator_func时,括号中的’func’指代被装饰器装饰的函数,在这段代码中指代的就是my_func函数。

  • 在被装饰器装时候,函数的实际执行执行顺序变成了内部函数wrapper所指定的顺序。即先执行print(“Before calling the function”);再执行func()指代的my_func函数;最后执行print(“After calling the function”)。

  • 本段代码的最终输出为:Before calling the function;Inside the function;After calling the function

3 使用装饰器记录函数执行次数

def cal_times(func):
   l=[]
   def wrapper(*var):
       l.append('1')
       func(*var)
       print("函数执行了%s次"%(len(l)))
   return wrapper
@cal_times
def my_func(i):
   print('%s的平方是%s'%(i,i**2))
my_func(5)
my_func(6)

my_func(i)函数中,我们增加了形参的输入,因此,在装饰器中,也要为之做出更改。此处装饰器中的wrapper函数我们使用*var传参,这种设计方式的优点是可以让这个装饰器适用于任何函数。

再加入了cal_times装饰器后,函数每运行一次,都会使列表l添加一个1,这样可以计算函数的运行次数。这段代码的运行结果如下:

Python装饰器使用方法全面梳理

4 带参数的装饰器

装饰器与函数一样,也可以带入参数,我们在第二节的基础上,对代码做出如下修改:

def decorator_func(param1, param2):
   def decorator(func):
       def wrapper():
           print("Before calling the function with params:", param1,param2)
           func()
           print("After calling the function")
       return wrapper
   return decorator
@decorator_func("hello", "world")
def my_func():
   print("Inside the function")
my_func()

这段代码使用了装饰器来在my_func函数调用前后打印额外的信息,并且装饰器函数decorator_func接受两个参数,在调用wrapper函数时使用这两个参数。

最终,代码将输出:

Before calling the function with params: hello world; Inside the function;After calling the function。

如果同时还有字典类型的参数传入,可以使用(*var,**_var)进行解决

5 装饰器处理有返回值的函数

前面我们定义的函数都是执行某种功能,不涉及到return的相关操作。当涉及到处理有返回值的函数时,对于内部函数我们应该使用一个变量将函数的运行结果保存起来,并放在内层函数的return中。为了实现这一功能,我们将第三部分的代码做出如下修改:

def cal_times(func):
   l=[]
   def wrapper(*var):
       l.append('1')
       result = func(*var)
       print("函数执行了%s次"%(len(l)))
       return result
   return wrapper
@cal_times
def my_func(i):
   print('%s的平方是%s'%(i,i**2))
   return i**2
a = my_func(5)
b = my_func(6)
print(a,b)

对于这个装饰器,我们在内部函数wrapper使用result保存运行结果,并将result return,这样a与b就可以被正常的赋值,运行结果如下图。

Python装饰器使用方法全面梳理

而如果不执行保存result并return,a和b将不会得到任何值:

Python装饰器使用方法全面梳理

来源:https://blog.csdn.net/nkufang/article/details/128761627

标签:Python,装饰器
0
投稿

猜你喜欢

  • 对python中大文件的导入与导出方法详解

    2021-02-11 12:32:47
  • 浅谈Pycharm中的Python Console与Terminal

    2021-09-19 07:20:42
  • VS 2008的性能改进

    2007-10-07 21:42:00
  • Python中json.load()和json.loads()有哪些区别

    2022-11-09 09:24:05
  • python可视化实现KNN算法

    2022-07-23 11:17:03
  • Python相互导入的问题解决

    2022-12-04 16:59:56
  • Python中OpenCV实现简单车牌字符切割

    2023-09-19 18:53:59
  • 如何恢复MYSQL的ROOT口令

    2008-06-02 13:59:00
  • Anaconda安装后Spyder闪退解决办法

    2023-04-18 10:17:01
  • 浅谈配置OpenCV3 + Python3的简易方法(macOS)

    2023-03-26 02:35:36
  • python爬虫之基金信息存储

    2021-12-25 03:20:04
  • Python实现ElGamal加密算法的示例代码

    2023-05-23 15:25:58
  • 基于Python实现简易的动漫图片转换器

    2022-12-25 23:23:11
  • Windows下在CMD下执行Go出现中文乱码的解决方法

    2024-04-25 15:17:27
  • Python3读写ini配置文件的示例

    2023-05-29 22:59:21
  • 如何在Python中创建二叉树

    2022-07-30 06:27:35
  • MySQL修改默认字符集

    2010-11-02 12:11:00
  • 基于python计算滚动方差(标准差)talib和pd.rolling函数差异详解

    2023-04-09 17:28:45
  • php实现通过cookie换肤的方法

    2023-11-23 17:57:07
  • 将Python代码打包为jar软件的简单方法

    2023-10-25 17:46:34
  • asp之家 网络编程 m.aspxhome.com