装饰器入门

在不修改程序源代码和程序调用方式的情况下,扩展程序功能时不得不用到装饰器。

python中的装饰器可谓功能强大,强大到刚接触它就被它弄得措手不及。

但是,静下心来好好研究,那可是回味无穷。于是,整理了下面2个demo,归纳总结。

demo1:简单装饰器

  1. #demo1
  2. def w(func):
  3. def inner():
  4. print("begin...")
  5. func()
  6. print("end...")
  7. return inner
  8. @w #语法糖:@w
  9. def test():
  10. print("我就是那个被装饰的函数")
  11. test()
  1. #output:
  2. begin...
  3. 我就是那个被装饰的函数
  4. end...

这就是一个简单的装饰器。所谓装饰指的是在test函数功能的基础上增加一些别的功能,本demo中用两次打印(即:begin...\end...)代替,且调用的方式不变还是原来的 test()。

初看demo1你应该是不明白到底发生了什么,那个@w是什么意思。

其实,@w 放在test函数的定义上面,这种简单的语法是python装饰器的一种简写。这样显的简单大气上档次。

但是,最原始、最本质的其实是高阶函数的使用,涉及到知识点有:

  • 函数可以当参数传给另一个参数
  • 函数可以当返回值被返回给一个变量,调用该变量就相当于调用被返回的函数

说了这么多,还是不如直接看代码来的快。

  1. #demo1本质
  2. def w(func):
  3. print("我要开始装饰啦...")
  4. def inner():
  5. print("begin...")
  6. func()
  7. print("end...")
  8. return inner
  9. def test():
  10. print("我就是那个被装饰的函数")
  11. test=w(test) #本质所在
  12. test()

你看,test=w(test) 这行语句是多么强大,玩的多么巧妙。

首先,我们应该知道这行语句执行的本质:

  1. 把test函数名当实参传给w函数,传给w函数中的形参func,
  2. 另外,w函数中嵌套定义了inner函数,func在inner中被调用,被装饰的test函数在inner中被调用,
  3. 但是 inner函数在w中没有被调用而是当返回值被返回到w函数外。接收的变量是test,此处的test仅仅是以一个变量名,虽然和需要被装饰的函数test长的一模一样,但本质是不一样的。
  4. 然后,程序的最后调用 test(),本质不是调用需要被装饰的那个test函数,而是调用了w函数的返回的函数 inner。
  5. 于是,直到现在被装饰的函数test,才在inner调用被执行中间调用。

这就是装饰的全部过程。总结:使用test=w(test)之后,调用test(),就相当于调用了inner()。

所以现在明白了demo1的output输出结果的顺序了吗?

因为首先调用的是w函数,然后遇到w函数返回inner函数,再开始调用inner函数。

现在回头看,@w放在test函数的定义前,其实很简单很方便地隐藏了私底下那些“见不得人”的操作。

不过这样的好处是显而易见的,官方取名为 语法糖,简单形象。

demo2:demo1补充

为了加深装饰器的本质执行流程,还有必要再唠叨两句,直接看代码。

  1. #demo2:
  2. def w(func):
  3. print("我要开始装饰啦....") #w中第一条语句
  4. def inner(): #w中第二条语句
  5. print("begin...")
  6. func()
  7. print("end...")
  8. print("好了装饰完毕了吗?") #w中第三条语句
  9. return inner
  10. @w #语法糖:@w
  11. def test():
  12. print("我就是那个被装饰的函数")
  13. test()

你猜,输出结果会是神马样子,是你想的那个样子吗?

我还是直接说吧,其实我在代码中加了注释的。

我们知道,装饰器的本质执行语句是:test=w(test)

  • 首先,是执行w函数,那就从上往下,分别还行语句1、语句2、语句3。。。
  • 所以,首先执行的是语句1,打印 :我要开始装饰啦....
  • 然后,语句2(其实这样说不准确),但是语句2是函数的定义,在w中并未被调用,所以此时并不会有输出,
  • 再其次是语句3,打印:好了装饰完毕了吗?
  • 于是w函数执行完毕了,遇到了return返回,返回的是inner函数名给test这个变量名
  • 最后的最后,调用test加(),即执行inner函数,于是打印:begin... 等

所以打印结果如下:

  1. #output:
  2. 我要开始装饰啦....
  3. 好了装饰完毕了吗?
  4. begin...
  5. 我就是那个被装饰的函数
  6. end...

demo3:两个装饰器

一个函数被两个装饰器装饰,是检验你是否理解装饰器的好东西。

  1. #demo3:
  2. def w1(func):
  3. print("我是w1")
  4. def inner():
  5. print("w1,begin...")
  6. func()
  7. print("w1,end...")
  8. return inner
  9. def w2(func):
  10. print("我是w2")
  11. def inner():
  12. print("w2,begin...")
  13. func()
  14. print("w2,end...")
  15. return inner
  16. @w2
  17. @w1
  18. def test():
  19. print("我是那个被装饰的函数")
  20. test()

像上面的思路一样,我们来捋一遍。

demo3中两个语法糖叠加的效果等同于:test=w2(w1(test))

因为两个装饰器w1、w2里面都有参数func和inner函数,可能容易混乱。

所以,下面叙述中提到的func1指的是w1函数的func,inner2指的是w2的inner,其他类似。

  1. #1 按照从左到右的执行顺序,test=w2(w1(test)),首先是把需要被装饰的test函数当实参传给w1函数中的形参func1;
  2. #2 w1函数就会执行,执行相关语句知道遇到return语句返回inner1,此时外面还有一个装饰器,inner1就不是直接返回给一个变量名,而是返回给w2当参数,即func2接收实参inner1;
  3. #3 w2函数就会执行,执行相关语句知道遇到return语句返回inner2,给变量test;
  4. #4 调用加括号的test,就相当于调用inner2(),此时inner2中的func2是inner1,因为#2中func2接收的实参是inner1呀;
  5. #5 当func2加括号调用时,就是调用inner1,此时inner1中的func1是需要被装饰的test,因为#1中func1接收的实参是test呀。
  6. 总结:
  7. 调用test()执行步骤如下:
  8. step1:执行w1,func1形参接收实参test,开始执行语句,先输出“我是w1”,再返回inner1w2
  9. step2:执行w2,func2形参接收实参inner1,开始执行语句,先输出“我是w2”,再返回inner2test变量
  10. step2:执行test(),就是执行inner2(),于是输出“w2,begin”,调用func2(),此即调用inner1,
  11. step3:输出“w1.begin”,再调用func1,即调用需要被装饰的test函数,输出“我是那个被装饰的函数”,
  12. step4:然后依次输出,“w1,end
  13. step5:此时inner2()中的func2执行完毕,再执行print("w2,end..."),输出“w2,end...”。
  1. #output:
  2. 我是w1
  3. 我是w2
  4. w2,begin...
  5. w1,begin...
  6. 我是那个被装饰的函数
  7. w1,end...
  8. w2,end...

总结:

  1. #test(w2(w1(test)))
  2. test()
  • 多个装饰器装饰,靠近被装饰函数的先执行
  • w1装饰器中:func是被装饰的函数test,
  • w2装饰器中:func是w1的返回值inner,
  • 最后,调用 变量test(),即执行w2的inner()

python3-三个demo带你入门装饰器的更多相关文章

  1. 08 . Python3高阶函数之迭代器、装饰器

    Python3高阶函数之迭代器.装饰器 列表生成式 推导式就是构建比较有规律的列表,生成器. 孩子,我现在有个需求,看列表[0, 1, 2, 3, 4, 5, 6, 7, 8, 9],我要求你把列表里 ...

  2. Python带参数的装饰器

    在装饰器函数里传入参数 # -*- coding: utf-8 -*- # 2017/12/2 21:38 # 这不是什么黑魔法,你只需要让包装器传递参数: def a_decorator_passi ...

  3. 学以致用三十六-----弄懂python装饰器

    看了海峰老师讲解的装饰器视频,讲解的非常棒.根据视频,记录笔记如下: 装饰器: 1.本质是函数,用def来定义.功能就是用来(装饰)其他函数,为其他函数添加附加功能 现有两个函数如下, def tes ...

  4. Python 带参数的装饰器 [2] 函数参数类型检查

    在Python中,不知道函数参数类型是一个很正常的事情,特别是在一个大项目里.我见过有些项目里,每一个函数体的前十几行都在检查参数类型,这实在是太麻烦了.而且一旦参数有改动,这部分也需要改动.下面我们 ...

  5. python 全栈开发,Day12(函数的有用信息,带参数的装饰器,多个装饰器装饰一个函数)

    函数的执行时,*打散.函数的定义时,*聚合. from functools import wraps def wrapper(f): # f = func1 @wraps(f) def inner(* ...

  6. Python之函数的进阶(带参数的装饰器)

    函数篇--装饰器二 带参数的装饰器 def outer(flag): def timer(func): def inner(*args,**kwargs): if flag: print('''执行函 ...

  7. Python_函数的有用信息、带参数的装饰器、多个装饰器装饰一个函数

    函数的有用信息 代码1: def login(username, password): """ 此函数需要用户名,密码两个参数,完成的是登录的功能. :return: T ...

  8. python:带参数的装饰器,函数的有用信息

    一.带参数的装饰器,函数的有用信息 def func1(): '''此函数的功能是完成的登陆的功能 return: 返回值是登陆成功与否(true,false) ''' print(333) func ...

  9. python全栈开发day12-函数的有用信息、带参数的装饰器、多个装饰器装饰一个函数、global和nonlocal的进一步解析和总结

    1.上周回顾 1).函数名的应用 直接打印函数名,是函数的地址 变量 函数的参数 函数的返回值 可以当容器类数据类型的元素 2).闭包 内层函数对外层函数的非全局变量的引用,就是闭包. 并返回内部函数 ...

随机推荐

  1. window下 nginx 80端口被占用

    问题:启动nginx没有反应,查看日志提示 bind() to 0.0.0.0:80 failed (10013: An attempt was made to access a socket in ...

  2. CSS-03 queue方法

    queue方法 摘自W3C school手册,用于简单理解使用queue方法 队列 每个元素均可拥有一到多个由 jQuery 添加的函数队列.在大多数应用程序中,只使用一个队列(名为 fx).队列运行 ...

  3. 微信小程序(3)--页面跳转和提示框

    微信小程序页面跳转方法: 1.<navigator url="../test/test"><button>点我可以切换可以返回</button> ...

  4. K8S命令大总结

    一.k8s-kubectl命令大全 Kubectl命令行管理对象类型 命令 描述 基础命令 create 通过文件名或标准输入创建资源. expose 将一个资源公开为一个新的Kubernetes服务 ...

  5. HDU4035 Maze 期望DP+树形DP(好题)

    题意:有一个树形的迷宫,有N个房间(标号为1~N)以及N-1条通道将它们连通,一开始在1号房间,每进入一个房间i,有k[i]的概率被陷阱杀死回到房间1,有s[i]的概率找到出口逃离迷宫,如果没有找到出 ...

  6. c++消息中间件

    ZeroMQ ActiveMQ-CPP 另外 ZeroMQ 的作者用 C 重构了一套.改名叫:nanomsg ZeroMQ:https://www.cnblogs.com/rainbowzc/p/33 ...

  7. windows和mtu值修改

    前言 有时候我们需要修改mtu值来对付乱七八糟的网络问题 windows修改方法 1.netsh interface ipv4 show subinterfaces 查询到目前系统的MTU值 2.ne ...

  8. bzoj4244 & loj2878. 「JOISC 2014 Day2」邮戳拉力赛 括号序列+背包

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=4244 https://loj.ac/problem/2878 题解 挺妙的一道题. 一开始一直 ...

  9. Kettle解析JSON错误,We MUST have the same number of values for all paths,We can not find and data with path [$.

    最近公司要从聚石塔上抽取数据,其中有JSON格式数据,所以学习一下Kettle解析JSON,碰到小小问题,记录一下: (1) 2015/07/15 15:22:48 - trade_detail.0 ...

  10. 英语单词custom

    custom 来源——xshell快捷键 翻译 n. 习惯,惯例:风俗:海关,关税:经常光顾:[总称](经常性的)顾客 adj. (衣服等)定做的,定制的 高中 | 初中 词源 英语单词custom含 ...