第6章 函数

6.1 函数的定义和调用

6.2 参数传递

6.3 函数返回值

6.4 变量作用域

6.5 匿名函数(lambda)

6.6 递归函数

6.7 迭代器

6.8 生成器

6.9 装饰器

6.9 装饰器

装饰器本质上是一个 Python 函数或类,它可以让其他函数或类在不需要做任何代码修改的前提下增加额外功能,它也不会改变函数或类的调用方式。

它经常用于有切面需求的场景,比如:插入日志、性能测试、事务处理、缓存、权限校验等场景,装饰器是解决这类问题的绝佳设计。有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码到装饰器中并继续重用。

概括的讲,装饰器的作用就是为已经存在的对象添加额外的功能。

装饰器的语法是将@装饰器名,放在被装饰对象上面。

有两类装饰器:函数装饰器、类装饰器

# 函数装饰器的语法
@dec
def func():
pass
参数说明:
@称为语法糖
dec:装饰器名字
func:被装饰的函数名

在介绍装饰器之前,先明确几个概念及原则:

原则:

Python程序是从上往下顺序执行的,而且碰到函数的定义代码块是不会立即执行的,只有等到该函数被调用时,才会执行其内部的代码块。

如果一个函数定义了两次,那么后面定义的会覆盖前面的定义。

在Python中代码的放置位置是有区别的,不能随意摆放,通常函数体要放在调用的语句之前。

概念:

函数名、函数体、返回值,函数的内存地址、函数名加括号、函数名被当作参数、函数名加括号被当作参数、返回函数名、返回函数名加括号。

# 加操作:无参数
import time def timer(func):
def inner():
t1 = time.time()
func()
t2 = time.time()
return t2 - t1 return inner @timer
def add():
a = 1
b = 1
time.sleep(1)
c = a + b
print(c)
return c add = add()
print(add)

函数名: timeraddinner

函数体:除def那一行外的部分

返回值: return后面的表达式

函数的内存地址:id(add)id(outer)等等

函数名加括号:对函数进行调用,比如add()timer(func)

函数名作为参数: timer(func)中的func本身是个函数,但作为参数被传递给了timer函数

函数名加括号被当做参数:其实就是先调用函数,再将它的返回值当做别的函数的参数,例如timer(add())

返回函数名:return inner

返回函数名加括号:return inner(),其实就是先执行inner函数,再将其返回值作为别的函数的返回值。

一个装饰器例子:

'''
1、@timer,称为语法糖,其作用是将被装饰函数add的函数名作为实参传给装饰器timer函数的形参func,传函数名实际传的是内存指向;
2、函数timer内又嵌套了inner函数,返回值是函数名inner,必须要明确的是,传进去函数名add,返回来inner函数名,结果就是add和inner共同指向在内存中定义inner的起始地址,而此时形参func指向在内存定义add的起始地址,可在调试程序过程中通过id方法打印查看add和inner所指向的内存地址的变化情况。
明确这两点,装饰器就不难理解了。
'''
# 加操作:无参数
import time def timer(func):
def inner():
t1 = time.time()
func()
t2 = time.time()
return t2 - t1 return inner @timer
def add():
a = 1
b = 1
time.sleep(1)
c = a + b
print(c)
return c add = add()
print(add)
# 加操作:位置参数
import time def timer(func):
def inner(a, b):
t1 = time.time()
func(a, b)
t2 = time.time()
return t2 - t1 return inner @timer
def add(a, b):
# a = 1
# b = 1
time.sleep(1)
c = a + b
print(c)
return c add = add(1, 1)
print(add)
# 加操作:可变参数
import time def timer(func):
def inner(*args):
t1 = time.time()
func(*args)
t2 = time.time()
return t2 - t1 return inner @timer
def add(a, b):
# a = 1
# b = 1
time.sleep(1)
c = a + b
print(c)
return c add = add(1, 1)
print(add)
# 加减乘除操作:可变参数
import time def timer(func):
def inner(*args):
t1 = time.time()
func(*args)
t2 = time.time()
return t2 - t1 return inner @timer
def add(a, b):
# a = 1
# b = 1
time.sleep(1)
c = a + b
print(c)
return c @timer
def plus(a, b):
# a = 1
# b = 1
time.sleep(2)
c = a - b
print(c)
return c @timer
def multi(a, b):
# a = 1
# b = 1
time.sleep(3)
c = a * b
print(c)
return c @timer
def div(a, b):
# a = 1
# b = 1
time.sleep(4)
c = a / b
print(c)
return c add = add(1, 1)
print(add)
plus = plus(2, 1)
print(plus)
multi = multi(3, 1)
print(multi)
div = div(4, 1)
print(div)

敬请关注个人微信公众号:测试工匠麻辣烫

Python3 装饰器解析的更多相关文章

  1. 净心诀---python3装饰器

    python3装饰器 装饰器作用 简单理解:可以为已有函数添加额外功能 例: 已有2个函数如下 def MyFunc1(): print("This is a print function1 ...

  2. JS 装饰器解析

    随着 ES6 和 TypeScript 中类的引入,在某些场景需要在不改变原有类和类属性的基础上扩展些功能,这也是装饰器出现的原因. 装饰器简介 作为一种可以动态增删功能模块的模式(比如 redux ...

  3. python3 装饰器应用举例

    [引子] python 中的装饰器是oop(面向对象编程)设计模式.之装饰器模式的一个应用.由于有语法糖衣的缘故.所以写起来也更加方便 [从一个比较经典的应用场景来讲解装饰器] 有过一定编程经历的工程 ...

  4. python3 装饰器全解

    本章结构: 1.理解装饰器的前提准备 2.装饰器:无参/带参的被装饰函数,无参/带参的装饰函数 3.装饰器的缺点 4.python3的内置装饰器 5.本文参考 理解装饰器的前提:1.所有东西都是对象( ...

  5. python3 装饰器

    #Author by Andy#_*_ coding:utf-8 _*_#装饰器的原则及构成:# 原则:# 1.不能修改被装饰函数的源代码.# 2.不能修改被装饰函数的调用方式.# 3.不能改变被装饰 ...

  6. python3 装饰器初识 NLP第三条

    还是先抄一条NLP假设... 三,有效果比有道理更重要   光说做法有道理或者正确而不顾是否有效果,是在自欺欺人. 在三赢(我好,人好,世界好)的原则基础上追求效果,比坚持什么是对的更有意义. 说道理 ...

  7. python3装饰器用法示例

    装饰器在编写后台的逻辑时有可能会用到,比方说一个场景:公司的员工想要登录自己公司的考勤记录系统去修改自己的考勤,以前是随便谁都有权限去修改,这样老板不同意了,现在,要在你登录前加一个权限验证的逻辑,如 ...

  8. python3装饰器

    由于函数也是一个对象,而且函数对象可以被赋值给变量,所以,通过变量也能调用该函数. >>> def now(): ... print('2015-3-25') ... >> ...

  9. python3 装饰器修复技术@wraps到底是什么?

    Python 装饰器中的@wraps的作用:    装饰器的作用:    在不改变原有功能代码的基础上,添加额外的功能,如用户验证等    @wraps(view_func)的作用:     不改变使 ...

随机推荐

  1. Beta冲刺--项目测试

    这个作业属于哪个课程 软件工程 (福州大学至诚学院 - 计算机工程系) 这个作业要求在哪里 Beta 冲刺 这个作业的目标 Beta冲刺--项目测试 作业正文 如下 其他参考文献 ... Beta冲刺 ...

  2. 第四模块 :微服务调用链监控CAT架构和实践

    采样率:每一个请求为都进行记录,或者100次请求为记录50次 各个开源框架都满足opentracing的标准,只要使用opentracing标准埋点的客户端,可以使用不同的客户端去展示,opentra ...

  3. python 模块 来了 (调包侠 修炼手册一)

    模块 什么是模块 模块:就是一系列功能的结合体 ,也可以说 一个.py文件包含了 Python 对象定义和Python语 那么 他就 可以说是 一个模块 模块的三种来源: 1.内置的(python解释 ...

  4. element ui 版本升级

    element ui 版本升级 1. 卸载之前版本 npm uninstall element-ui 2.重新安装element-ui npm i element-ui 3.就如package.jso ...

  5. SSTI-服务端模板注入漏洞

      原理: 服务端模板注入是由于服务端接收了用户的输入,将其作为 Web 应用模板内容的一部分,在进行目标编译渲染的过程中,执行了用户插入的恶意内容,因而导致了敏感信息泄露.代码执行.GetShell ...

  6. apply()方法和call()介绍

    我们发现apply()和call()的真正用武之地是能够扩充函数赖以运行的作用域. 1.call,apply都属于Function.prototype的一个方法,它是JavaScript引擎内在实现的 ...

  7. JavaScript中__proto__与prototype的关系(转)

    一.所有构造器/函数的__proto__都指向Function.prototype,它是一个空函数(Empty function) 1 2 3 4 5 6 7 8 9 Number.__proto__ ...

  8. Fetch方法封装、业务实践

    说Fetch之前啊,我们不得不说一说Ajax了,以前使用最多的自然是jQuery封装的Ajax方法了,强大而且好用. 有人说了,jQuery的Ajax都已经封装得那么好了,你还整Fetch干什么,这不 ...

  9. window的常用操作

    一.window.location location对象属性 1.location.href 属性返回当前页面的 URL. 2.location.pathname 返回路径和方法名称 3.locati ...

  10. BZOJ3573 米特运输 题解

    题目 米特是D星球上一种非常神秘的物质,蕴含着巨大的能量.在以米特为主要能源的D星上,这种米特能源的运输和储存一直是一个大问题.D星上有N个城市,我们将其顺序编号为1到N,1号城市为首都.这N个城市由 ...