Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Python装饰器的具体使用

Python装饰器的具体使用

原创
作者头像
忆想不到的晖
修改于 2021-04-27 06:21:19
修改于 2021-04-27 06:21:19
6040
举报
文章被收录于专栏:huihui

装饰器分类

Python装饰器,大致可分为:无参装饰器、带参装饰器。接下来我们一探究竟

多个装饰器一起使用

代码语言:txt
AI代码解释
复制
"""
装饰器的具体使用
"""

print("# -------------------- 多个装饰器一起使用 -------------------- #")


# 加粗
def make_bold(fn):
    def wrapped():
        return "<b>" + fn() + "</b>"

    return wrapped


# 斜体
def make_italic(fn):
    def wrapped():
        return "<i>" + fn() + "</i>"

    return wrapped


@make_bold
def test1():
    return "hello world-1"


@make_italic
def test2():
    return "hello world-2"


# 两个装饰器一起使用
@make_bold
@make_italic
def test3():
    return "hello world-3"


print(test1())
print(test2())
print(test3())

运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 多个装饰器一起使用 -------------------- #
<b>hello world-1</b>
<i>hello world-2</i>
<b><i>hello world-3</i></b>

可以发现装饰 test3 的结果是先变斜体,然后在加粗。首先程序是从上到下执行的,当遇到 @make_bold 时它会把下面的函数引用传递给 make_bold 函数,但下面的又是一个装饰器 @make_italic ,这个装饰器一样会把下面的函数 test3 传递给 make_italic 函数,先将make_italic 函数的返回值 wrapped 赋值给 test3,然后这个新的 test3 函数再传递给 make_bold。因此是先变斜体,然后再加粗。

可能理解起来会很绕,看看下面这段程序结果你就更明白了,最好是亲自去尝试一下,加强理解。

代码语言:txt
AI代码解释
复制
"""
装饰器的具体使用
"""

print("# -------------------- 多个装饰器一起使用 -------------------- #")


# 加粗
def make_bold(fn):
    print("make_bold called")
    print(fn.__name__)

    def wrapped():
        return "<b>" + fn() + "</b>"

    return wrapped


# 斜体
def make_italic(fn):
    print("make_italic called")
    print(fn.__name__)

    def wrapped():
        return "<i>" + fn() + "</i>"

    return wrapped


# 两个装饰器一起使用
@make_bold
@make_italic
def test3():
    return "hello world-3"


print(test3())

运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 多个装饰器一起使用 -------------------- #
make_italic called
function name: test3
    
make_bold called
function name: wrapped
<b><i>hello world-3</i></b>

先传递 test3 函数,经过 @make_italic 装饰器后 test3 = wrapped,在传递 test3 函数,而此时test3 其实就是 wrapped。因此第二次 fn.__name__,打印是 wrapped

各种装饰器示例

无参数的函数

代码语言:txt
AI代码解释
复制
print("# -------------------- 无参数装饰器 -------------------- #")
from time import ctime, sleep


# 打印当前运行时间
def cur_time(func):
    def wrapped_func():
        print("%s called at %s" % (func.__name__, ctime()))
        func()

    return wrapped_func


@cur_time
def foo():
    print("I am foo")


foo()
sleep(2)
foo()

上面代码理解装饰器执行行为可理解成

代码语言:txt
AI代码解释
复制
# foo先作为参数赋值给func后,foo接收指向timefun返回的wrapped_func
foo = timefun(foo)

# 调用foo(),即等价调用wrapped_func()
foo()

# 内部函数wrapped_func被引用,所以外部函数的func变量(自由变量)并没有释放
# func里保存的是原foo函数对象

运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 无参数装饰器 -------------------- #
foo called at Thu Apr 15 20:40:35 2021
I am foo
foo called at Thu Apr 15 20:40:37 2021
I am foo

被装饰的函数有参数

代码语言:txt
AI代码解释
复制
print("# -------------------- 被装饰的函数有参数 -------------------- #")
from time import ctime, sleep


def cur_time(func):

    # 注意在这里接受函数参数
    def wrapped_func(a, b):
        print("%s called at %s" % (func.__name__, ctime()))
        print(a, b)
        func(a, b)

    return wrapped_func


@cur_time
def foo(a, b):
    print(a + b)


foo(3, 5)
sleep(2)
foo(2, 4)

运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 被装饰的函数有参数 -------------------- #
foo called at Thu Apr 15 20:47:18 2021
3 5
8
foo called at Thu Apr 15 20:47:20 2021
2 4
6

被装饰的函数有不定长参数

代码语言:txt
AI代码解释
复制
print("# -------------------- 被装饰的函数有不定长参数 -------------------- #")
from time import ctime, sleep


def cur_time(func):

    # 这里用*arg **kwargs来接受不定长参数
    def wrapped_func(*args, **kwargs):
        print("%s called at %s" % (func.__name__, ctime()))
        print("args:", args)
        print("kwargs:", kwargs)
        func(*args, **kwargs)

    return wrapped_func


@cur_time
def foo(a, b, c):
    print(a + b + c)


foo(1, 3, 5)
sleep(2)
foo(2, 4, c=6)

运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 被装饰的函数有不定长参数 -------------------- #
foo called at Thu Apr 15 21:10:02 2021
args: (1, 3, 5)
kwargs: {}
9

foo called at Thu Apr 15 21:10:05 2021
args: (2, 4)
kwargs: {'c': 6}
12

foo(2, 4, c=6) 举例,2, 4 会以元组的形式传递给 argsc=6 这种键值对的则是以字典的形式传递给 kwargs ,需注意的是在函数参数定义的时候需在参数名前带上 *。参数名称可以自定义但 * 不能少,否则就不是不定参数函数。例如 *args 可以换成 *params*kwargs 可以换成 *kws。但我们最好不要改动。

装饰器中的return

代码语言:txt
AI代码解释
复制
print("# -------------------- 装饰器的return -------------------- #")
from time import ctime, sleep


def cur_time(func):
    def wrapped_func():
        print("%s called at %s" % (func.__name__, ctime()))
        func()

    return wrapped_func


@cur_time
def foo():
    print("I am foo")


@cur_time
def get_info():
    return '----hei hei---'


foo()
sleep(2)
foo()

print(get_info())

执行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 装饰器的return -------------------- #
foo called at Thu Apr 15 21:27:52 2021
I am foo

foo called at Thu Apr 15 21:27:54 2021
I am foo

get_info called at Thu Apr 15 21:27:54 2021
None

如果修改装饰器为如下

代码语言:txt
AI代码解释
复制
def cur_time(func):
    def wrapped_func():
        print("%s called at %s" % (func.__name__, ctime()))
        ret = func() # 先接收函数的返回值
        return ret   # 然后返回出去

    return wrapped_func

则运行结果:

代码语言:txt
AI代码解释
复制
# -------------------- 装饰器的return -------------------- #
foo called at Thu Apr 15 21:32:43 2021
I am foo

foo called at Thu Apr 15 21:32:45 2021
I am foo

get_info called at Thu Apr 15 21:32:45 2021
----hei hei---

说明:

一般情况下为了让装饰器更通用,可以有 return

装饰器带参数

代码语言:txt
AI代码解释
复制
print("# -------------------- 装饰器带参数 -------------------- #")
from time import ctime, sleep


def time_arg(pre="hello"):

    def cur_time(func):
        def wrapped_func():
            print("%s called at %s %s" % (func.__name__, ctime(), pre))
            return func()

        return wrapped_func

    return cur_time


@time_arg("hui")
def foo():
    print("I am foo")


@time_arg("python")
def goo():
    print("I am goo")


foo()
sleep(2)
foo()

goo()
sleep(2)
goo()

装饰过程如下

代码语言:txt
AI代码解释
复制
1. 先调用 time_arg("hui")

2. 将步骤1得到的返回值,即cur_time返回, 然后装饰器@time_arg('hui')就变成了@cur_time

3. 将 foo 传递给 cur_time

3. 将 cur_time(foo)的结果返回,即wrapped_func

4. 让foo = wrapped_fun,即foo现在指向wrapped_func

可以理解为

代码语言:txt
AI代码解释
复制
foo() == timef_arg("hui")(foo)()
goo() == timef_arg("python")(goo)()

记住 xxx()xxx 调用,而 @decorate 则执行 decorate 函数,只不过会把 @ 下面被装饰的函数当做参数传递给 decorate 函数执行。

类装饰器(扩展)

装饰器函数其实是这样一个接口约束,它必须接受一个 callable 对象作为参数,然后返回一个 callable 对象。在Python中一般 callable 对象都是函数,但也有例外。只要某个对象重写了 __call__() 方法,那么这个对象就是 callable 的。

  • callable 就是可被调用执行的对象
代码语言:txt
AI代码解释
复制
class Test():
    def __call__(self):
        print('call me!')

t = Test()
t()  # 输出 call me

类装饰器demo

代码语言:txt
AI代码解释
复制
class Test(object):
    
    def __init__(self, func):
        print("---初始化---")
        print("func name is %s" % func.__name__)
        self.__func = func
        
    def __call__(self):
        print("---装饰器中的功能---")
        self.__func()

@Test
def test():
    print("----test---")
test()

说明:

代码语言:txt
AI代码解释
复制
1. 当用Test来装作装饰器对test函数进行装饰的时候,首先会创建Test的实例对象
	并且会把test这个函数名当做参数传递到__init__方法中
    即在__init__方法中的属性__func指向了test指向的函数

2. test指向了用Test创建出来的实例对象

3. 当在使用test()进行调用时,就相当于让这个对象(),因此会调用这个对象的__call__方法

4. 为了能够在__call__方法中调用原来test指向的函数体,
	所以在__init__方法中就需要一个实例属性来保存这个函数体的引用
    所以才有了self.__func = func这句代码,从而在调用__call__方法中能够调用到test之前的函数体

运行结果如下:

代码语言:txt
AI代码解释
复制
---初始化---
func name is test
---装饰器中的功能---
----test---

源代码

源代码已上传到 Gitee PythonKnowledge: Python知识宝库,欢迎大家来访。

✍ 码字不易,还望各位大侠多多支持❤️。

公众号

新建文件夹X

大自然用数百亿年创造出我们现实世界,而程序员用几百年创造出一个完全不同的虚拟世界。我们用键盘敲出一砖一瓦,用大脑构建一切。人们把1000视为权威,我们反其道行之,捍卫1024的地位。我们不是键盘侠,我们只是平凡世界中不凡的缔造者 。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
深入理解 Python 中的装饰器
装饰器本质上也是函数,接受函数对象来作为参数,并在装饰器的内部来调用接受的函数对象完成相关的函数调用。也可以这样理解,为了方便在几个不同函数调用之前或者之后完成相关的统一操作,注意是完成统一的操作,可以传参数使得装饰器不完全一样,后面会讲到。最重要的应用如工程应用上记录相关的内部调用接口的流水日志,不同的接口需要统一的样式,所以可以用装饰器来实现。先简单看一下示例:
测试小兵
2019/11/19
8900
python装饰器Decorators
http://blog.csdn.net/pipisorry/article/details/41902599
用户7886150
2021/01/19
3260
我是装饰器
其实,我并不难理解,而且学会使用我之后,可以让你写代码时偷点懒,少点重复性工作,代码也更优雅,更具有 Pythonic。
somenzz
2020/11/25
3600
python第二十六课——装饰器
装饰器是闭包的一种使用场景; python中的装饰器在定义上需要传入一个函数对象, 在此函数执行之前或者之后都可以追加其它的操作, 这样做的好处是,在不改变源码(原本业务逻辑的)同时,进行功能的扩展; 它在python中一般被使用在,性能测试,插入日志,事务管理,权限校验... 它就好比是一个切面(可插拔的),也就是我们之后学习中会提到的叫面向切面编程(aop) 开放封闭原则: 开放: 在不改动源码(破坏原本业务逻辑)的同时扩展新的功能 封闭: 不允许随意去修改源代码 说明装饰器的好处: 部门A:维护和管理数据信息平台 信息平台:内部封装了一些核心的api和接口 装饰器:函数(fn) 部门B: m1(): func1() func2() func3() 部门C: func4() func5() func6()
hankleo
2020/09/16
4040
没看完这11 条,别说你精通 Python 装饰器
对于每一个学习 Python 的同学,想必对 @ 符号一定不陌生了,正如你所知, @ 符号是装饰器的语法糖,@符号后面的函数就是我们本文的主角:装饰器。
小小詹同学
2019/08/16
9770
没看完这11 条,别说你精通 Python 装饰器
Python进阶之强大的装饰器 Decorators (二)
当然,我们也可以同时使用多个装饰器。 def my_logging(func): def wrapper(): print('logging - {} is running'.format(func.__name__)) func() # run func() Equivalent run f1() return wrapper def bold(func): def wrapper(): print("<b>")
用户4945346
2020/06/16
3100
Python每日一题:装饰器(完整篇)
要理解装饰器,你首先必须要知道在Python中,函数是对象。这一点对装饰器有着很重要的影响。让我们用一个简单的例子来看一下为什么:
用户7685359
2020/08/22
1.1K0
【从零学习python 】33.装饰器的作用(二)
hello world-1 hello world-2 hello world-3
全栈若城
2024/02/29
1360
Python装饰器的详细解析
Python装饰器(fuctional decorators)就是用于拓展原来函数功能的一种函数,目的是在不改变原函数名(或类名)的情况下,给函数增加新的功能。
用户8949263
2022/04/08
5760
Python 工匠:使用装饰器的技巧
装饰器(Decorator) 是 Python 里的一种特殊工具,它为我们提供了一种在函数外部修改函数的灵活能力。它有点像一顶画着独一无二 @ 符号的神奇帽子,只要将它戴在函数头顶上,就能悄无声息的改变函数本身的行为。
崔庆才
2019/07/22
5450
Python 两种装饰器
装饰器(Decorators)是 Python 的一个重要部分。简单地说:他们是修改其他函数的功能的函数。他们有助于让我们的代码更简短,也更Pythonic(Python范儿)。 
不吃西红柿
2022/07/29
2180
Python装饰器
如果想在一个函数执行前后执行一些别的代码,比如打印一点日志用来输出这个函数的调用情况那应该怎么做呢?
职场亮哥
2020/10/10
4840
让代码更具 Python 范儿的装饰器
在 Python 中,装饰器的作用是在不改变函数或类的代码的前提下,改变函数或类的功能。在介绍装饰器之前,我们先来复习下 Python 中的函数。
mr.songw
2021/01/14
4350
让代码更具 Python 范儿的装饰器
python高级-装饰器(19)
某公司有多个研发部⻔,1个基础平台部⻔,基础平台负责提供底层的功能,如:数据库操作、redis调⽤、监控API等功能。研发部⻔使⽤基础功能时,只需调⽤基础平台提供的功能即可。如下:
Se7eN_HOU
2019/09/11
4200
python高级-装饰器(19)
Python装饰器是什么?
如果想在一个函数执行前后执行一些别的代码,比如打印一点日志用来输出这个函数的调用情况那应该怎么做呢?
科技新语
2025/01/17
1020
Python装饰器是什么?
装饰器
装饰器Decorators是Python的重要组成部分。 简而言之:它们是修改另一个函数功能的函数。 他们有助于使我们的代码更简洁,更Pythonic。
Helloted
2022/06/07
3460
Python(三)对装饰器的理解
装饰器是 Python 的一个重要部分,也是比较难理解和使用好的部分。下面对装饰器做一下简单整理 1. 前言 装饰器实际上是应用了设计模式里,装饰器模式的思想: 在不概念原有结构的情况下,添加新的功能 类似于我们穿不同的衣服,可以先穿一件衬衫,再穿一件毛衣,再穿一件羽绒服 但是毛衣不会影响羽绒服,羽绒服也不会影响衬衫 随时更换,同一个人可以有不同的穿衣打扮 对比之下,每一个装饰器就代表上述的一件衣服,我们可以根据功能需求,给一个函数本身加上不同的外套,也可以调整外套之间的顺序 装饰器本质上就是一个个的
西凉风雷
2022/11/23
4470
python『学习之路03』装饰器
#!/usr/bin/env python # -*- coding: utf-8 -*- # @Time : 2017/11/19 10:45 # @Author : mixiu26 # def foo(): # print("in the foo") # bar() ---->> bar() 方法未定义 # foo() # # def bar(): # print("in the bar()") # def foo(): # print("in the foo()") # bar() ---- >> 正常运行 # foo() # 改进: 内存加载时机是先定义在调用, 函数也是变量,所以呢,代码运行逻辑是从上到下,就是在调用foo()之前, 就先在内存中定义了 # 变量foo,就是将方法体赋给foo, 这里其实什么也没有做, 然后定义bar()这个函数,然后在调用运行foo()这个函数,而在运行它之前 # foo() 和 bar() 这个方法已经存在了, 变量的使用规则就是,先定义在使用, 这里也一样适用 --- >> 函数即'变量' # def foo(): # print("in the foo()") # bar() # ---- >> 正常运行 # def bar(): # print("in the bar()") # foo() # 改进: # def foo(): # print("in the foo()") # bar() # ---- >> 无法正常运行 --- >>因为foo()调用前 bar()还未定义 # foo() # def bar(): # print("in the bar()") # 高阶函数:=========================>> # 1.吧一个函数当做实参传递给另外一个函数: # 2.返回值中包含函数名 # def bar(): # print("in the bar()") # def test1(func): # print(func) # bar的内存地址: <function bar at 0x0000000002452E18> # func() # in the bar() # # test1(bar) # in the bar() # 解释, 首先函数即是变量不解释, 关于变量的使用 ---- >> x = 1; y = x; ----- >> y = 1 # 我们调用test1() --- >>传入了bar ---- 这里的bar == func ---- >>其实就是把bar的内存地址给了func , # 因为bar在内存中作为一个变量来存储方法体, test1中吧bar传给func, 那么这里的func == bar了 --- 他们指向的方法体是一样的只不过在加了个func的引用而已 import time # 函数作为实参传递的高阶函数 # def bar(): # time.sleep(3) # print("in the bar()") # in the bar() # # def test1(func): # start_time = time.time() # func() # sttop_time = time.time() # print("the func run time is %s" %(sttop_time-start_time)) # the func run time is 3.010805130004883 # test1(bar) # in the bar() # 返回值中携带函数的高阶函数: # def bar(): # time.sleep(3) # print("in the bar") # # def test2(func): # print(func) # return func # 注意传值问题: # test2(bar()) ---- >> 这样传值就不符合高阶函数定义, 是要把函数作为实参传递, 如果你带了括号传递的就是bar() 这个方法的返回值 # 而传递: test(bar) ---- >> 这里的bar 传递的就是地址值, 是这个方法体的引用, 所以传递的时候一定要注意 # test2(bar) # 当我们吧bar传过来, test2就开始打印bar的内存地址, 打印完成后又返回bar的内存地址, # 我们将test2的运行结果进行接收: 注意了,我们知道, 函数即是变量. so --- >>ret() <=
呆呆
2021/05/18
3810
python装饰器1:函数装饰器详解
假如你已经定义了一个函数funcA(),在准备定义函数funcB()的时候,如果写成下面的格式:
py3study
2020/01/19
7480
六、面向对象进阶
生成器 1、什么是生成器 生成器是这样一个函数,它记住上一次返回时在函数体中的位置。对生成器函数的第二次(或第 n 次)调用跳转至该函数中间,而上次调用的所有局部变量都保持不变。 生成器不仅“记住”了它数据状态;生成器还“记住”了它在流控制构造(在命令式编程中,这种构造不只是数据值)中的位置。 生成器的特点: 节约内存 迭代到下一次的调用时,所使用的参数都是第一次所保留下的,即是说,在整个所有函数调用的参数都是第一次所调用时保留的,而不是新创建的 2. 创建生成器方法1 要创建一个生成器,有很多种方法。第一
酱紫安
2018/04/16
6080
相关推荐
深入理解 Python 中的装饰器
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档