装饰器


一、定义

1.装饰器:本质是函数

2.功能:用来装饰其他函数,为其他函数添加附加功能

二、原则

1.不能修改被装饰函数的源代码

2.不能修改被装饰函数的调用方式

三、实现装饰器

1.函数 即 变量 的概念

2.高阶函数

3.嵌套函数

>> 高阶函数 + 嵌套函数 = 装饰器

四、函数 即 变量

1、函数和变量的类比

  1. x = 1
  2. print(id(x))
  3.  
  4. def test():
  5. pass
  6.  
  7. print(test)
  8.  
  9. #输出
  10.  
  11. 1842348496
  12. <function test at 0x0000020DBA24D2F0>

  在上例中我们定义了一个变量 "x" 和一个函数 test(),我们分别打印出变量和函数在内存中的位置。可以看出,print(test) 即 print("函数名")的时候,我们可以打印出函数的内存地址。

  我们看下面的代码:

  1. def test():
  2. print("in the test.")
  3.  
  4. f=test
  5. f()
  6.  
  7. #输出
  8. in the test.

  我们把函数名 test 赋予给 f,然后运行 f(),可以看出函数是可以正常运行的,而且就是test函数的运行结果。那么这和下面的代码是不是类似:

  1. x = 1
  2. y = x
  3.  
  4. print(y)
  5.  
  6. #输出
  7. 1

  我们可以做出如下类比,函数名 test 相当于 x ,函数体就相当于 1,而 f 就相当于 y。

2.python中内存的表现形式

   

  我们把绿方块当作是内存,每一个小方块就是变量或者函数在内存当中的地址。而变量名(x)或者函数名(test),我们可以形象的将他们比作门牌号。当需要调用变量或者函数的时候,我们只要引用他们的门牌号就可以找到他们的内存地址并返回。只是函数的运行需要加(),如 test()。

  既然调用变量其实就是引用变量的内存地址,而调用函数名同样可以得到函数体的内存地址。我们就可以把函数名当作变量名传给函数,即函数就是变量。而将函数当作参数的函数,也就是高阶函数。

五、高阶函数

满足下列条件之一就是高阶函数

1.把一个函数名当作实参传给另外一个函数

2.返回值中包含函数名

1)函数名作为参数

  1. import time
  2.  
  3. def test1():
  4. time.sleep(2)
  5. print("in the test1.")
  6.  
  7. def test2(func):
  8. start_time = time.time()
  9. func()
  10. stop_time = time.time()
  11. print("The action time of program is {}".format(stop_time-start_time))
  12.  
  13. test2(test1)
  14.  
  15. #输出
  16. in the test1.
  17. The action time of program is 2.0012054443359375

  以上事例中,我们定义了一个函数 test1,同时也定义了一个高阶函数test2。我们把test1函数名当作参数传入test2中,可以实现这样一个功能,为原本的test1函数添加了一个计算运行时间的功能。这有点像装饰器了,但是有一点符合,就是上面的高阶函数test2改变了函数的调用方式。

  但是我们实现了在不修改被装饰函数的基础上,添加了新功能。

2)返回值中有函数名

  1. def test1():
  2. time.sleep(2)
  3. print("in the test1.")
  4.  
  5. def test2(func):
  6. print(func)
  7. return func
  8.  
  9. test1 = test2(test1)
  10. test1()
  11.  
  12. #输出
  13. <function test1 at 0x000001E5D853D2F0>
  14. in the test1.

  在上例中,我们最后将高阶函数test2(test1) 赋予给了test1,再次调用test1函数。我们可以直观的看到test1函数的调用方式在此例中没有改变。但是也并没有添加新功能,而这就需要使用到嵌套函数了。

六、嵌套函数

在函数体内又另一个函数的完整定义,这就是嵌套函数。

1)定义:

  1. def foo():
  2. print("in the foo")
  3. def bar():
  4. print("in the bar")
  5.  
  6. bar()
  7.  
  8. foo()

仅仅在函数内容调用函数,就不是嵌套函数,如下:

  1. def test1():
  2. print("in the test1.")
  3.  
  4. def test2():
  5. test1()

2)嵌套函数的作用域

局部作用域和全局作用域的访问顺序

  1. x = 0
  2. def grandpa():
  3. x = 1
  4. def dad():
  5. x = 2
  6. def son():
  7. x = 3
  8. print(x)
  9. son()
  10. dad()
  11. grandpa()
  12.  
  13. #输出
  14. 3

3)使用嵌套函数为被修饰函数添加新功能 

  在高阶函数第二例中,我们实现了不改变原函数的调用方式。而需要添加新功能的话,就要求修饰内容存在在返回值中,即return func 中,我们可以定义一个嵌套函数来实现这个功能。

  1. import time
  2.  
  3. def timer(func): # timer(test1) func = test1
  4. def deco():
  5. start_time = time.time()
  6. func() # run test1()
  7. stop_time = time.time()
  8. print("the action time of the program is {}".format(stop_time-start_time))
  9. return deco # 返回了deco的内存地址
  10.  
  11. def test1():
  12. time.sleep(2)
  13. print("in the test1.")
  14.  
  15. test1 = timer(test1)
  16. test1()
  17. # 输出
  18. in the test1.
  19. the action time of the program is 2.0003786087036133

  我们在timer()内部定义了一个嵌套函数 deco(),这个嵌套函数实现了为被修饰函数添加运行时间的功能。而timer()函数返回了deco()的内存地址,这个内存地址deco就可以被引用,甚至直接赋予给 test1。这样我们就可以直接运行 test1(),这样就实现我们的装饰器的功能。

七、装饰器 

  python通过在函数定义前添加一个装饰器名和@符号,来实现对函数的包装

  1. import time
  2.  
  3. def timer(func): # timer(test1) func = test1
  4. def deco():
  5. start_time = time.time()
  6. func() # run test1()
  7. stop_time = time.time()
  8. print("the action time of the program is {}".format(stop_time-start_time))
  9. return deco # 返回了deco的内存地址
  10.  
  11. @timer # test1 = timer(test1)
  12. def test1():
  13. time.sleep(2)
  14. print("in the test1.")
  15.  
  16. test1()

  

  

装饰器--decorator1的更多相关文章

  1. Python 学习笔记9(装饰器,decorator)

    31 装饰器 装饰器可以对一个函数.方法或者类进行加工,是一种高级的python语法. 装饰函数 接收一个可调用对象作为输入参数,并返回一个新的可调用对象. 把函数传递给装饰器,然后增加新的功能,返回 ...

  2. python装饰器小计

    1.装饰器:本质是函数,是用来给其他函数添加附加扩展功能的函数,其返回值是一个函数(函数指针) 2.装饰器作用:不改变函数源代码和函数调用方式的前提下添加函数的附加功能. 3.装饰器储备知识点: A. ...

  3. JAVA装饰器模式

    Java程序员们应该对java.io对不会陌生,因为java.io包采用了装饰器模式. 一.定义: Decorator装饰器,顾名思义,就是动态地给一个对象添加一些额外的职责,就好比为房子进行装修一样 ...

  4. 如何理解Python装饰器

    如何理解Python装饰器?很多学员对此都有疑问,那么上海尚学堂python培训这篇文章就给予答复. 一.预备知识 首先要理解装饰器,首先要先理解在 Python 中很重要的一个概念就是:“函数是 F ...

  5. python装饰器1:函数装饰器详解

    装饰器1:函数装饰器 装饰器2:类装饰器 装饰器3:进阶 先混个眼熟 谁可以作为装饰器(可以将谁编写成装饰器): 函数 方法 实现了__call__的可调用类 装饰器可以去装饰谁(谁可以被装饰): 函 ...

  6. Javascript装饰器的妙用

    最近新开了一个Node项目,采用TypeScript来开发,在数据库及路由管理方面用了不少的装饰器,发觉这的确是一个好东西.装饰器是一个还处于草案中的特性,目前木有直接支持该语法的环境,但是可以通过 ...

  7. Python学习系列之装饰器

    装饰器的作用 装饰器用于装饰某个函数.方法或者类,它可以让这个函数执行之前或者执行之后做一些操作 手工实现一个装饰器 def outer(some_func): #装饰器 $1 def inner() ...

  8. Python学习笔记之生成器、迭代器和装饰器

    这篇文章主要介绍 Python 中几个常用的高级特性,用好这几个特性可以让自己的代码更加 Pythonnic 哦 1.生成器 什么是生成器呢?简单来说,在 Python 中一边循环一边计算的机制称为 ...

  9. python第四周:装饰器、迭代器、内置方法、数据序列化

    1.装饰器 定义:本质是一个函数,(装饰其他函数)就是为其他函数添加附加功能 原则:不能修改被装饰函数的源代码,不能修改被装饰函数的调用方式 实现装饰器的知识储备: 函数即“变量”.每当定义一个函数时 ...

随机推荐

  1. PPM、PGM、PBM图像格式剖析

    今天突然需要用到PPM这个图像文件格式,之前没见过,在此记录一下. PPM.PGM.PBM这三个图像文件格式很少见,其实也不难,分别用于彩色图像.灰度图像.二值图像.这里以PPM格式为例. PPM格式 ...

  2. .NET 4.0 Tuple 元组

    Tuple是.NET 4.0的新特性,主要功能是动态返回数据结构,也可以用做临时数据结构. 原来做一些功能时需要一个方法返回几个值,有两种方法: 1. 非常难看.难用的OUT参数: 2. 新写一个实体 ...

  3. rhel6 mysql skip-grant-tables 添加用户报错 ERROR 1290

    不小心把数据库密码忘掉了, 这个时候我们只需要在数据库的配置文件里面添加 skip-grant-tables 然后重新启动服务,再登录数据库就不要我们输入密码了 这个时候我成功登录数据,可是不小心又把 ...

  4. Python爬虫入门(1-2):综述、爬虫基础了解

    大家好哈,最近博主在学习Python,学习期间也遇到一些问题,获得了一些经验,在此将自己的学习系统地整理下来,如果大家有兴趣学习爬虫的话,可以将这些文章作为参考,也欢迎大家一共分享学习经验. Pyth ...

  5. 遗传算法框架GAFT优化小记

    前言 前段时间一直在用自己写的遗传算法框架测试算法在优化力场参数的效果,但是跑起来效率很慢,因为适应度函数需要调用多次力场程序计算能量,但是还是比我预想中的慢我也没有及时对程序进行profiling和 ...

  6. 关于虚拟机安装mac os 教程详解

    环境搭建 VMware下载 百度云盘下载:链接:http://pan.baidu.com/s/1pK8RcLl 密码:5jc5 Unlocker208 百度云盘下载:链接:http://pan.bai ...

  7. ACM 第二十天

    积性函数.杜教筛 练习题 莫比乌斯函数之和 51Nod - 1244 莫比乌斯函数,由德国数学家和天文学家莫比乌斯提出.梅滕斯(Mertens)首先使用μ(n)(miu(n))作为莫比乌斯函数的记号. ...

  8. Hibernate:工作原理

    Hibernate的工作原理图如下所示:

  9. HTTPD解析介绍

    配置文件全局介绍 (1)主配置文件:/etc/httpd/conf/httpd.conf 全局配置:Section 1: Global Environment 第33行 中心主机配置: Section ...

  10. 3dContactPointAnnotationTool开发日志(三一)

      在玩的时候遇到了一个python的问题: Traceback (most recent call last): File ".\convert.py", line 13, in ...