前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >一则小故事带你弄清Python装饰器

一则小故事带你弄清Python装饰器

原创
作者头像
忆想不到的晖
修改于 2021-05-07 02:15:27
修改于 2021-05-07 02:15:27
4520
举报
文章被收录于专栏:huihui

装饰器

装饰器是程序开发中经常会用到的一个功能,用好了装饰器,开发效率如虎添翼,所以这也是Python面试中必问的问题。装饰器 (Decorators) 是 Python 的一个重要部分。简而言之:Python中的装饰器就是拓展原来函数功能的一种函数

回顾函数引用

代码语言:txt
AI代码解释
复制
#### 第一波 ####
def func():
    print("func() called")

func    # 表示是函数
func()  # 表示执行func函数

#### 第二波 ####
def func():
    print("func() called")

func = lambda x: x + 1

func()  # 执行lambda表达式,而不再是原来的func函数,因为func这个名字被重新指向了另外一个匿名函数

从上一文 深入浅出Python闭包 中,就知道函数名仅仅是个变量,只不过指向了定义的函数而已,所以才能通过 函数名() 调用,如果 函数名 = xxx 被修改了,那么当在执行 函数名() 时,调用的就不知之前的那个函数了。

这里又提到 lambda 表达式,其就是一个 匿名函数。下面一图就可参透

lambda表达式图解
lambda表达式图解

小故事带你了解装饰器的功能

初创公司有N个业务部门,基础平台部门负责提供底层的功能,如:数据库操作、redis调用、监控API等功能。业务部门使用基础功能时,只需调用基础平台提供的功能即可。如下:

代码语言:txt
AI代码解释
复制
############### 基础平台提供的功能如下 ###############

def f1():
    print('f1')

def f2():
    print('f2')

def f3():
    print('f3')

def f4():
    print('f4')

############### 业务部门A 调用基础平台提供的功能 ###############

f1()
f2()
f3()
f4()

############### 业务部门B 调用基础平台提供的功能 ###############

f1()
f2()
f3()
f4()

目前公司有条不紊的进行着,但是,以前基础平台的开发人员在写代码时候没有关注验证相关的问题,即:基础平台的提供的功能可以被任何人使用。现在需要对基础平台的所有功能进行重构,为平台提供的所有功能添加验证机制,即:执行功能前,先进行验证

老大把工作交给 Low B,他是这么做的:

跟每个业务部门交涉,让每个业务部门自己写代码,调用基础平台的功能之前先验证。诶,这样一来基础平台就不需要做任何修改了。太棒了,有充足的时间泡妹子...

当天Low B 被开除了…

老大把工作交给 Low BB,他是这么做的:

代码语言:txt
AI代码解释
复制
############### 基础平台提供的功能如下 ############### 

def f1():
    # 验证1
    # 验证2
    # 验证3
    print('f1')

def f2():
    # 验证1
    # 验证2
    # 验证3
    print('f2')

def f3():
    # 验证1
    # 验证2
    # 验证3
    print('f3')

def f4():
    # 验证1
    # 验证2
    # 验证3
    print('f4')

############### 业务部门不变 ############### 
### 业务部门A 调用基础平台提供的功能### 

f1()
f2()
f3()
f4()

### 业务部门B 调用基础平台提供的功能 ### 

f1()
f2()
f3()
f4()

过了一周 Low BB 被开除了…

老大把工作交给 Low BBB,他是这么做的:

只对基础平台的代码进行重构,其他业务部门无需做任何修改

代码语言:txt
AI代码解释
复制
############### 基础平台提供的功能如下 ############### 

# 新增一个验证函数
def check():
    # 验证1
    # 验证2
    # 验证3
    pass

def f1():
    check()
    print('f1')

def f2():
    check()
    print('f2')
    
def f3():
    check()
    print('f3')

def f4():
    check()
    print('f4')
    

老大看了下Low BBB 的实现,嘴角漏出了一丝的欣慰的笑,语重心长的跟Low BBB聊了个天:

老大说:

写代码要遵循 开放封闭 原则,虽然在这个原则是用的面向对象开发,但是也适用于函数式编程,简单来说,它规定已经实现的功能代码不允许被修改,但可以被扩展,即:

  • 封闭:已实现的功能代码块
  • 开放:对扩展开发

如果将开放封闭原则应用在上述需求中,那么就不允许在函数 f1、f2、f3、f4 的内部进行修改代码,老板就给了Low BBB一个实现方案:

代码语言:txt
AI代码解释
复制
# 这个就是闭包应用,只不过传递过来的 func 是函数的引用
def check(func):
    def inner():
        # 验证1
        # 验证2
        # 验证3
        func()
    return inner

@check
def f1():
    print('f1')
    
@check
def f2():
    print('f2')
    
@check
def f3():
    print('f3')
    
@check
def f4():
    print('f4')

对于上述代码,也是仅仅对基础平台的代码进行修改,就可以实现在其他人调用函数 f1, f2, f3, f4 之前都进行【验证】操作,并且其他业务部门无需做任何操作。

Low BBB心惊胆战的问了下,这段代码的内部执行原理是什么呢?

老大正要生气,突然Low BBB的手机掉到地上,恰巧屏保就是Low BBB的女友照片,老大一看一紧一抖,喜笑颜开,决定和Low BBB交个好朋友。

详细的开始讲解了:

单独以 f1() 为例:

代码语言:txt
AI代码解释
复制
def check(func):
    def inner():
        # 验证1
        # 验证2
        # 验证3
        func()
    return inner

@check
def f1():
    print('f1')

Python解释器就会从上到下解释代码,步骤如下:

  1. def check(func): ==> 将check函数加载到内存
  2. @check

没错, 从表面上看解释器仅仅会解释这两句代码,因为函数在 没有被调用之前其内部代码不会被执行。

从表面上看解释器着实会执行这两句,但是 @check 这一句代码里却有大文章, @函数名是Python的一种语法糖。

上例@check内部会执行一下操作:

执行check函数

执行check函数 ,并将 @check 下面的函数作为 check函数的参数

即:@check 等价于 check(f1) 所以内部就会去执行:

代码语言:txt
AI代码解释
复制
def inner(): 
    #验证 1
    #验证 2
    #验证 3
    func()    # func是参数,此时 func 等于 f1 
return inner

返回的 inner,inner代表的是函数,非执行函数 ,其实就是将原来的 f1 函数塞进另外一个函数中

check函数的返回值

将执行完的check函数返回值 赋值 给@check下面的函数的函数名f1 即将check的返回值再重新赋值给 f1,即:

代码语言:txt
AI代码解释
复制
新f1 = def inner(): 
            # 验证 1
            # 验证 2
            # 验证 3
            原来f1()
        return inner

所以,以后业务部门想要执行 f1 函数时,就会执行 新f1 函数,在新f1 函数内部先执行验证,再执行原来的f1函数,然后将原来f1 函数的返回值返回给了业务调用者。

如此一来, 即执行了验证的功能,又执行了原来f1函数的内容,并将原f1函数返回值 返回给业务调用着

Low BBB 你明白了吗?要是没明白的话,我晚上去你家帮你解决吧!!!

这则生动形象的小故事是引用网上的。

Python 装饰器简单应用

Python中装饰器的语法以 @ 开头,接着是装饰器函数的名字、可选参数。

紧跟装饰器声明的是被装饰的函数和被装饰的函数的可选参数,如下:

代码语言:txt
AI代码解释
复制
@decorator(装饰器的可选参数)
def func(函数参数):
	......
    

计算函数运行时间

寻找0, 1000之间三个数,条件: a + b + c = 1000,且符合a*a + b*b = c*c, 结果不能重复,

如(0, 500, 500) 和 (500, 0, 500)只能出现一个,注:a, b, c 就是要寻找的三个数。

代码语言:txt
AI代码解释
复制
"""
Python 重点知识装饰器
"""
import time


print("# -------------------- 计算函数运行时间 -------------------- #")

def calc_time(func):
    """
    计算函数运行时间
    """
    def calc():
        # 函数调用前,记录开始时间
        start_time = time.time()
        func()
        # 函数结束,计算运行时间
        use_time = time.time() - start_time
        print(func.__name__, "use time {} seconds".format(use_time))

    return calc


@calc_time
def fun1():
    """
    寻找[0, 1000]之间三个数
    条件: a + b + c = 1000,
         且符合a*a + b*b = c*c,
         结果不能重复
    """
    ret_set = set()
    for i in range(0, 1001):
        for j in range(0, 1001):
            for m in range(0, 1001):
                if (i + j + m) == 1000 and (i*i + j*j) == m*m:
                    # print(i, j, m)
                    li = [i, j, m]
                    
                    # 记得排序后再去添加到集合中
                    # 防止[0, 500, 500]、[500, 0, 500]等不去重
                    li.sort()   
                    ret_set.add(tuple(li))

    print(ret_set)
    

@calc_time
def fun2():
    """
    改进版
    """
    ret_set = set()
    for i in range(10001):
        for j in range(1001 - i):
            m = 1000 - i - j
            if i*i + j*j == m*m:
                # print(i, j, m)
                li = [i, j, m]
                li.sort()
                ret_set.add(tuple(li))
    print(ret_set)


def main():
    fun1()
    print()
    fun2()


if __name__ == '__main__':
    main()
    

运行结果

代码语言:txt
AI代码解释
复制
# -------------------- 计算函数运行时间 -------------------- #
{(0, 500, 500), (200, 375, 425)}
fun1 use time 99.87939643859863 seconds

{(0, 500, 500), (200, 375, 425)}
fun2 use time 0.12499618530273438 seconds

这段程序不仅说明了装饰器的强大,可复用性高,还提醒大家设计一个好程序,程序性能倍翻。

源代码

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

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

公众号

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

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

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

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

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

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
python装饰器的学习
在python中,装饰器是一种增加函数功能的简单方法,利用装饰器功能可以很快的给不同的函数插入相同的功能。
无涯WuYa
2018/10/25
4180
python装饰器的学习
Python进阶之强大的装饰器 Decorators (一)
这篇文章主要介绍 decorator(装饰器),在开始介绍 decorator 前,要先有一个观念,就是在 python 中,函数是对象,可以将它们分配给变量和传递给其他函数并从其他函数返回,可以在其他函数中定义函数,并且子功能可以捕获父功能的本地状态。
用户4945346
2020/06/16
2900
Python函数装饰器基础知识
函数装饰器是Python语言最优秀的设计之一,它以非常简洁的方式增强了函数的行为,让崎岖不平之路变得平坦顺畅。
dongfanger
2022/05/09
2490
Python函数装饰器基础知识
【从零学习python 】32.装饰器的作用(一)
装饰器是程序开发中经常会用到的一个功能,用好了装饰器,开发效率如虎添翼,所以这也是Python面试中必问的问题。但对于好多初次接触这个知识的人来讲,这个功能有点绕,自学时直接绕过去了,然后面试问到了就挂了,因为装饰器是程序开发的基础知识,这个都不会,别跟人家说你会Python, 看了下面的文章,保证你学会装饰器。
全栈若城
2024/02/29
920
看完这篇文章还会不懂Python装饰器?掐死小编吧
1. 必备 #### 第一波 #### def foo(): print('foo') foo #表示是函数 foo() #表示执行foo函数 #### 第二波 #### def foo(): print('foo') foo = lambda x: x + 1 foo(1) # 执行下面的lambda表达式,而不再是原来的foo函数,因为函数 foo 被重新定义了 2. 需求来了 初创公司有N个业务部门,1个基础平台部门,基础平台负责提供底层的功能,如:数据库操作、r
崔庆才
2018/06/25
5810
看完这篇文章还不懂Python装饰器?
初创公司有N个业务部门,1个基础平台部门,基础平台负责提供底层的功能,如:数据库操作、redis调用、监控API等功能。业务部门使用基础功能时,只需调用基础平台提供的功能即可。如下:
QQ1622479435
2018/11/14
5080
Python3中的装饰器
def foo(): print("第一次定义的foo()函数") def foo(): print("第二次定义的foo()函数") foo() # 结果 第二次定义的foo()函数
py3study
2020/01/10
5910
Python随笔(四)、python基础
05 python s12 day4 迭代器原理及使用 什么是迭代: 可以直接作用于for循环的对象统称为可迭代对象(Iterable)。* 可以被next()函数调用并不断返回下一个值的对象称为迭代器(Iterator)。 所有的Iterable均可以通过内置函数iter()来转变为Iterator。 对迭代器来讲,有一个next()就够了。在你使用for 和 in 语句时,程序就会自动调用即将被处理的对象的迭代器对象,然后使用它的next()方法,直到监测到一个StopIteration异常。
py3study
2020/01/08
3360
Python随笔(四)、python基础
闭包和装饰器
在函数内部再定义⼀个函数,并且这个内部函数⽤到了外部的变量,这个函数以及⽤到外部函数的变量及参数叫闭包
@小森
2024/03/15
860
清清爽爽理解Python装饰器
不知不觉已经连续写了13天了,很享受每天写公众号这个过程,通过写作将自己每天学习的Python知识分享出来,但是学习的东西是零散的,需要自己通过一定的逻辑顺序将零散的知识串起来,这很锻炼自己的逻辑能力和写作能力。
stormwen
2019/08/05
3330
python高级-装饰器(19)
某公司有多个研发部⻔,1个基础平台部⻔,基础平台负责提供底层的功能,如:数据库操作、redis调⽤、监控API等功能。研发部⻔使⽤基础功能时,只需调⽤基础平台提供的功能即可。如下:
Se7eN_HOU
2019/09/11
4140
python高级-装饰器(19)
python装饰器和语法糖
装饰器在大工程中比较常见,那么如何理解装饰器呢?打个比方,假如你建好了一栋大房子,建好后还想加一些功能,这个时候房子的主体结构是不能动了,只好在现有房子的基础上做一些装饰/装修。这些装饰在不影响/不修改原来房子功能的基础上,增加了美观等功能。
烤粽子
2021/07/07
8070
装饰器进阶
装饰带参数函数 def foo(func): # 接收的参数是一个函数名 def bar(x, y): # 这里需要定义和被装饰函数相同的参数 print("这里是新功能...") # 新功能 func(x, y) # 被装饰函数名和参数都有了,就能执行被装饰函数了 return bar # 定义一个需要两个参数的函数 @foo def f1(x, y): print("{}+{}={}".format(x, y, x+y)) # 调用
新人小试
2018/04/12
6000
函数 之装饰器
  在看装饰器之前,我们先来搞清楚什么是闭包函数。python是一种面向对象的编程语言,在python中一切皆对象,这样就使得变量所拥有的属性,函数也同样拥有。这样我们就可以理解在函数内创建一个函数的行为是完全合法的。这种函数被叫做内嵌函数,这种函数只可以在外部函数的作用域内被正常调用,在外部函数的作用域之外调用会报错,例如:
全栈程序员站长
2022/07/21
4280
函数 之装饰器
深入了解Python中的装饰器
Python的装饰器是AOP编程的一种实现,其他很多语言也都支持装饰器模式。 注:AOP是指面向切面编程,详见 AOP概念
tunsuy
2022/10/27
3320
Python 函数装饰器和闭包
创建一个装饰器工厂函数,把参数传给它,返回一个装饰器,然后再把它应用到要装饰的函数上。
为为为什么
2022/08/09
6560
Python 函数装饰器和闭包
「Python」闭包与装饰器
请注意,本文编写于 1730 天前,最后修改于 999 天前,其中某些信息可能已经过时。
曼亚灿
2023/05/17
2100
python中的装饰器decorator
而我们想为这三个函数增加一个函数调用打印功能 类似print("call f1()")
py3study
2020/01/15
5520
一文读懂python装饰器由来(二)
上一篇文章主要以一步一步演进的方式介绍了装饰器的工作原理以及使用(没看的小伙伴可以关注一下 一文读懂Python装饰器由来(一)),其实只要认真学习上一篇文章,已经能够满足日常对装饰器的使用了。但是,若想真正理解装饰器,并进行更高阶的使用还要了解其他一些知识:
Python中文社区
2018/07/26
4240
python Function(函数)
函数是python为了代码最大程度地重用和最小化代码冗余而提供的基本程序结构。函数是一种设计工具,它能让程序员将复杂的系统分解为可管理的部件; 函数用于将相关功能打包并参数化。 在python中可以创建如下4种函数:     1)、全局函数:定义在模块中(直接定义在模块中的函数)。     2)、局部函数:嵌套于其它函数中(在函数中再定义的函数)。     3)、lambda函数:表达式。匿名函数(它仅是一个表达式),它可以出现在任何位置,很高的录活性。     4)、方法:与特定数据类型关联的函数,并且只能与数据类型相关一起使用。定义在类中的函数。    python也提供了很多内置函数 函数与过程的区别:     函数都有return返回值。返回一个对象 创建函数     def functionName(parameters):         suite 相关概念:     def 是一个可执行语句;因此可以出现在任何能够使用的地方,甚至可以嵌套于其它语句,例if或while中。def创建了一个对象  并将其赋值给一个变量名(即函数名);     return用于返回结果对象,其为可选项;无return语句的函数自动返回一个None对象;返回多个值时,彼此间使用逗号分隔,且组合为元组形式返回一个对象。     def语句运行之后,可以在程序中通过函数名后附加括号进行调用 。     例1:
py3study
2020/01/13
7830
相关推荐
python装饰器的学习
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档