Python语法详解之decorator装饰器

作者:GlassySky0816 时间:2021-07-15 23:11:45 

python 是一门优雅的语言,有些使用方法就像魔法一样。装饰器(decorator)就是一种化腐朽性为神奇的技巧。最近一直都在使用 Tornado 框架,一直还是念念不忘 Flask 。Flask 是我最喜欢的 Python 框架,最早被它吸引也是源自它使用装饰器这个语法糖(Syntactic sugar)来做 Router,让代码看上去就感觉甜甜的。

Tornado 中的 Router 略显平淡,怀念 Flask 的味道,于是很好奇的想知道 Flask 是如何使用这个魔法。通过阅读 Flask 的源码,我们也可以为 Tornado 实现了一个装饰器 Router。

当然对于刚接触 Python 的人,也许很容易理解装饰器本质是设计模式中的装饰器模式。可是 Python 通过@一个实现装饰器的语法糖。下面看下Python语法详解之decorator装饰器。

一、定义

装饰器 decorator 或者称为包装器,是对函数的一种包装。

二、作用

它能使函数的功能得到扩充,而同时不用修改函数本身的代码。它能够增加函数执行前、执行后的行为,而不需对调用函数的代码做任何改变。

三、举例

初始化函数


# 函数hello,输出 hello + name 的字符串
def hello(name):
   return 'hello ' + name

希望实现功能:在每一个调用 hello 函数的时候,将输出的字符串用 <tag>包住

例如:hello john 变成 <tag>hello john<tag>

方法一:自定义wrapper函数

这种方法成功修改了函数 hello 的行为,不过需要修改对 hello的调用。

每一个调用hello 的地方,都要给成调用wrapper,并修改参数列表


def hello(name):
   return 'hello ' + name

def wrapper(tag, func, *arg, **kvargs):
   tag = "<" + tag + ">"
   return tag + func(*arg, **kvargs) + tag

if __name__ == "__main__":
   print(wrapper('p', hello, 'john'))

输出

Python语法详解之decorator装饰器

方法二:自定义decorator函数

为了不改变对 hello的调用。我们需要得到一个新的函数对象,它修改 hello的行为,并用这个对象对 hello赋值。

从而调用 hello的时候,调用的是扩充行为后的 hello


def hello(name):
   return 'hello ' + name
def myDecorator(func, tag):
   def myWrapper(*arg, **kvargs):  # 重新包装func,其参数列表与func一致
       sign = "<" + tag + ">"
       return sign + func(*arg, **kvargs) + sign

return myWrapper

hello = myDecorator(hello, "div")  # 用新的函数对象修改hello

if __name__ == "__main__":
   print(hello("john"))

 这样,只要hello被myDecorator 赋值一次,以后再调用hello 时,就调用的是包装后的函数

输出

Python语法详解之decorator装饰器

方法三:python的decorator

python 的装饰器所做的事与方式2类似

它通过语法糖使装饰器看起来更清晰、简介,而不用每次都书写方式2中第7行代码 hello = myDecorator(hello, "div")


def setTag(tag):  # 由于此装饰器需要参数,所以要再套一层
   def myDecorator(func):  # 装饰器的核心,接受函数对象做参数,返回包装后的函数对象
       def myWrapper(*arg, **kvargs):  # 包装的具体过程
           sign = "<" + tag + ">"
           return sign + func(*arg, **kvargs) + sign

return myWrapper

return myDecorator

@setTag("div")  # 用@标签在定义函数时套上装饰器
def hello(name):
   return 'hello' + name

if __name__ == '__main__':
   print(hello('john'))

本质上,方式2 与 方式3 完成的是同一件事,只不过方式3 比方式2 代码更简洁,方便。

比如,现在要给 hello 函数套上三个标签<body><div><p>

如果用方式2


hello = myDecorator(myDecorator(myDecorator(hello, "body"),"div"),"p")

如果用方式3


@myDecorator("body")
@myDecorator("div")
@myDecorator("p")
def hello(name)
   return 'hello' + name

在多个装饰器嵌套的情况下,python内置的decorator 结构更清晰。

伪代码:


def myDecorator(...):#定义装饰器,可能带参数
   def decorator(func):    #装饰器核心,以被装饰的函数对象为参数,返回装饰后的函数对象
       def wrapper(*args, **kvargs):    #装饰的过程,参数列表适应不同参数的函数
            ...    #修改函数调用前的行为
            func(*args, **kvargs)    #调用函数
            ...    #修改函数调用后的行为
        return wrapper
    return decorator

@myDecorator(...):#给函数加上装饰器
def myFunc(...):  #自己定义的功能函数
   ...

知识点:

  • 在python中,当*和**符号出现在函数定义的参数中时,表示任意数目参数收集。*arg表示任意多个无名参数,类型为tuple;**kwargs表示关键字参数,为dict,使用时需将*arg放在**kwargs之前,否则会有“SyntaxError: non-keyword arg after keyword arg”的语法错误

  • 在函数调用时,*会以单个元素的形式解包一个元祖,使其成为独立的参数。

  • 在函数调用时,**会以键/值对的形式解包一个字典,使其成为独立的关键字参数。

来源:https://blog.csdn.net/qq_38784098/article/details/120873382

标签:Python,decorator,装饰器
0
投稿

猜你喜欢

  • MySQL 8.0.18 Hash Join不支持left/right join左右连接问题

    2024-01-27 20:51:13
  • 用python生成1000个txt文件的方法

    2023-03-24 07:51:11
  • Pycharm Available Package无法显示/安装包的问题Error Loading Package List解决

    2022-09-09 18:33:23
  • 动态给表添加删除字段并同时修改它的插入更新存储过程

    2024-01-17 22:12:56
  • 使用pyqt 实现重复打开多个相同界面

    2021-08-19 12:33:43
  • Python常用字符串替换函数strip、replace及sub用法示例

    2022-04-07 18:21:17
  • javascript的this关键字详解

    2024-04-17 10:08:44
  • 使用Abot中文分词组件来开发ASP站内搜索引擎

    2007-10-18 13:36:00
  • sqlserver获取当前日期的最大时间值

    2011-09-30 11:44:50
  • PHP中__LINE__,__FILE__,__DIR__等常用魔术常量实例讲解

    2023-06-13 23:42:32
  • 浅谈如何使用webpack构建多页面应用

    2024-04-16 09:27:13
  • mysql中TIMESTAMPDIFF案例详解

    2024-01-18 05:56:49
  • asp文章中随机插入网站版权文字的实现代码

    2011-04-15 11:11:00
  • Python Pandas 删除列操作

    2023-12-17 22:03:53
  • 数据库查询排序使用随机排序结果示例(Oracle/MySQL/MS SQL Server)

    2024-01-18 20:14:13
  • 使用access数据库时可能用到的数据转换

    2008-09-10 12:49:00
  • Sqlserver事务备份和还原的实例代码(必看)

    2024-01-23 18:44:14
  • Navicat Premium15连接云服务器中的数据库问题及遇到坑

    2024-01-19 05:41:50
  • Python多线程同步Lock、RLock、Semaphore、Event实例

    2023-08-03 20:47:15
  • PHP实现获取客户端IP并获取IP信息

    2023-06-18 08:34:26
  • asp之家 网络编程 m.aspxhome.com